12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471 |
- <?php
- /* For licensing terms, see /license.txt */
- use Chamilo\CoreBundle\Component\Utils\ChamiloApi;
- use Chamilo\CoreBundle\Entity\TrackEExercises;
- use ChamiloSession as Session;
- /**
- * Class ExerciseLib
- * shows a question and its answers.
- *
- * @author Olivier Brouckaert <oli.brouckaert@skynet.be> 2003-2004
- * @author Hubert Borderiou 2011-10-21
- * @author ivantcholakov2009-07-20
- * @author Julio Montoya
- */
- class ExerciseLib
- {
- /**
- * Shows a question.
- *
- * @param Exercise $exercise
- * @param int $questionId $questionId question id
- * @param bool $only_questions if true only show the questions, no exercise title
- * @param bool $origin i.e = learnpath
- * @param string $current_item current item from the list of questions
- * @param bool $show_title
- * @param bool $freeze
- * @param array $user_choice
- * @param bool $show_comment
- * @param bool $show_answers
- *
- * @throws \Exception
- *
- * @return bool|int
- */
- public static function showQuestion(
- $exercise,
- $questionId,
- $only_questions = false,
- $origin = false,
- $current_item = '',
- $show_title = true,
- $freeze = false,
- $user_choice = [],
- $show_comment = false,
- $show_answers = false,
- $show_icon = false
- ) {
- $course_id = $exercise->course_id;
- $exerciseId = $exercise->iId;
- if (empty($course_id)) {
- return '';
- }
- $course = $exercise->course;
- // Change false to true in the following line to enable answer hinting
- $debug_mark_answer = $show_answers;
- // Reads question information
- if (!$objQuestionTmp = Question::read($questionId, $course)) {
- // Question not found
- return false;
- }
- if ($exercise->getFeedbackType() != EXERCISE_FEEDBACK_TYPE_END) {
- $show_comment = false;
- }
- $answerType = $objQuestionTmp->selectType();
- $pictureName = $objQuestionTmp->getPictureFilename();
- $s = '';
- if ($answerType != HOT_SPOT &&
- $answerType != HOT_SPOT_DELINEATION &&
- $answerType != ANNOTATION
- ) {
- // Question is not a hotspot
- if (!$only_questions) {
- $questionDescription = $objQuestionTmp->selectDescription();
- if ($show_title) {
- if ($exercise->display_category_name) {
- TestCategory::displayCategoryAndTitle($objQuestionTmp->id);
- }
- $titleToDisplay = $objQuestionTmp->getTitleToDisplay($current_item);
- if ($answerType == READING_COMPREHENSION) {
- // In READING_COMPREHENSION, the title of the question
- // contains the question itself, which can only be
- // shown at the end of the given time, so hide for now
- $titleToDisplay = Display::div(
- $current_item.'. '.get_lang('ReadingComprehension'),
- ['class' => 'question_title']
- );
- }
- echo $titleToDisplay;
- }
- if (!empty($questionDescription) && $answerType != READING_COMPREHENSION) {
- echo Display::div(
- $questionDescription,
- ['class' => 'question_description']
- );
- }
- }
- if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION]) && $freeze) {
- return '';
- }
- echo '<div class="question_options">';
- // construction of the Answer object (also gets all answers details)
- $objAnswerTmp = new Answer($questionId, $course_id, $exercise);
- $nbrAnswers = $objAnswerTmp->selectNbrAnswers();
- $quizQuestionOptions = Question::readQuestionOption($questionId, $course_id);
- // For "matching" type here, we need something a little bit special
- // because the match between the suggestions and the answers cannot be
- // done easily (suggestions and answers are in the same table), so we
- // have to go through answers first (elems with "correct" value to 0).
- $select_items = [];
- //This will contain the number of answers on the left side. We call them
- // suggestions here, for the sake of comprehensions, while the ones
- // on the right side are called answers
- $num_suggestions = 0;
- switch ($answerType) {
- case MATCHING:
- case DRAGGABLE:
- case MATCHING_DRAGGABLE:
- if ($answerType == DRAGGABLE) {
- $isVertical = $objQuestionTmp->extra == 'v';
- $s .= '
- <div class="col-md-12 ui-widget ui-helper-clearfix">
- <div class="clearfix">
- <ul class="exercise-draggable-answer '.($isVertical ? '' : 'list-inline').'"
- id="question-'.$questionId.'" data-question="'.$questionId.'">
- ';
- } else {
- $s .= '<div id="drag'.$questionId.'_question" class="drag_question">
- <table class="data_table">';
- }
- // Iterate through answers
- $x = 1;
- //mark letters for each answer
- $letter = 'A';
- $answer_matching = [];
- $cpt1 = [];
- for ($answerId = 1; $answerId <= $nbrAnswers; $answerId++) {
- $answerCorrect = $objAnswerTmp->isCorrect($answerId);
- $numAnswer = $objAnswerTmp->selectAutoId($answerId);
- if ($answerCorrect == 0) {
- // options (A, B, C, ...) that will be put into the list-box
- // have the "correct" field set to 0 because they are answer
- $cpt1[$x] = $letter;
- $answer_matching[$x] = $objAnswerTmp->selectAnswerByAutoId(
- $numAnswer
- );
- $x++;
- $letter++;
- }
- }
- $i = 1;
- $select_items[0]['id'] = 0;
- $select_items[0]['letter'] = '--';
- $select_items[0]['answer'] = '';
- foreach ($answer_matching as $id => $value) {
- $select_items[$i]['id'] = $value['id_auto'];
- $select_items[$i]['letter'] = $cpt1[$id];
- $select_items[$i]['answer'] = $value['answer'];
- $i++;
- }
- $user_choice_array_position = [];
- if (!empty($user_choice)) {
- foreach ($user_choice as $item) {
- $user_choice_array_position[$item['position']] = $item['answer'];
- }
- }
- $num_suggestions = ($nbrAnswers - $x) + 1;
- break;
- case FREE_ANSWER:
- $fck_content = isset($user_choice[0]) && !empty($user_choice[0]['answer']) ? $user_choice[0]['answer'] : null;
- $form = new FormValidator('free_choice_'.$questionId);
- $config = [
- 'ToolbarSet' => 'TestFreeAnswer',
- ];
- $form->addHtmlEditor(
- "choice[".$questionId."]",
- null,
- false,
- false,
- $config
- );
- $form->setDefaults(["choice[".$questionId."]" => $fck_content]);
- $s .= $form->returnForm();
- break;
- case ORAL_EXPRESSION:
- // Add nanog
- if (api_get_setting('enable_record_audio') === 'true') {
- //@todo pass this as a parameter
- global $exercise_stat_info;
- if (!empty($exercise_stat_info)) {
- $objQuestionTmp->initFile(
- api_get_session_id(),
- api_get_user_id(),
- $exercise_stat_info['exe_exo_id'],
- $exercise_stat_info['exe_id']
- );
- } else {
- $objQuestionTmp->initFile(
- api_get_session_id(),
- api_get_user_id(),
- $exerciseId,
- 'temp_exe'
- );
- }
- echo $objQuestionTmp->returnRecorder();
- }
- $form = new FormValidator('free_choice_'.$questionId);
- $config = ['ToolbarSet' => 'TestFreeAnswer'];
- $form->addHtml('<div id="'.'hide_description_'.$questionId.'_options" style="display: none;">');
- $form->addHtmlEditor(
- "choice[$questionId]",
- null,
- false,
- false,
- $config
- );
- $form->addHtml('</div>');
- $s .= $form->returnForm();
- break;
- }
- // Now navigate through the possible answers, using the max number of
- // answers for the question as a limiter
- $lines_count = 1; // a counter for matching-type answers
- if ($answerType == MULTIPLE_ANSWER_TRUE_FALSE ||
- $answerType == MULTIPLE_ANSWER_COMBINATION_TRUE_FALSE
- ) {
- $header = Display::tag('th', get_lang('Options'));
- foreach ($objQuestionTmp->options as $item) {
- if ($answerType == MULTIPLE_ANSWER_TRUE_FALSE) {
- if (in_array($item, $objQuestionTmp->options)) {
- $header .= Display::tag('th', get_lang($item));
- } else {
- $header .= Display::tag('th', $item);
- }
- } else {
- $header .= Display::tag('th', $item);
- }
- }
- if ($show_comment) {
- $header .= Display::tag('th', get_lang('Feedback'));
- }
- $s .= '<table class="table table-hover table-striped">';
- $s .= Display::tag(
- 'tr',
- $header,
- ['style' => 'text-align:left;']
- );
- } elseif ($answerType == MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY) {
- $header = Display::tag('th', get_lang('Options'), ['width' => '50%']);
- echo "
- <script>
- function RadioValidator(question_id, answer_id)
- {
- var ShowAlert = '';
- var typeRadioB = '';
- var AllFormElements = window.document.getElementById('exercise_form').elements;
-
- for (i = 0; i < AllFormElements.length; i++) {
- if (AllFormElements[i].type == 'radio') {
- var ThisRadio = AllFormElements[i].name;
- var ThisChecked = 'No';
- var AllRadioOptions = document.getElementsByName(ThisRadio);
-
- for (x = 0; x < AllRadioOptions.length; x++) {
- if (AllRadioOptions[x].checked && ThisChecked == 'No') {
- ThisChecked = 'Yes';
- break;
- }
- }
-
- var AlreadySearched = ShowAlert.indexOf(ThisRadio);
- if (ThisChecked == 'No' && AlreadySearched == -1) {
- ShowAlert = ShowAlert + ThisRadio;
- }
- }
- }
- if (ShowAlert != '') {
-
- } else {
- $('.question-validate-btn').removeAttr('disabled');
- }
- }
-
- function handleRadioRow(event, question_id, answer_id) {
- var t = event.target;
- if (t && t.tagName == 'INPUT')
- return;
- while (t && t.tagName != 'TD') {
- t = t.parentElement;
- }
- var r = t.getElementsByTagName('INPUT')[0];
- r.click();
- RadioValidator(question_id, answer_id);
- }
-
- $(function() {
- var ShowAlert = '';
- var typeRadioB = '';
- var question_id = $('input[name=question_id]').val();
- var AllFormElements = window.document.getElementById('exercise_form').elements;
-
- for (i = 0; i < AllFormElements.length; i++) {
- if (AllFormElements[i].type == 'radio') {
- var ThisRadio = AllFormElements[i].name;
- var ThisChecked = 'No';
- var AllRadioOptions = document.getElementsByName(ThisRadio);
-
- for (x = 0; x < AllRadioOptions.length; x++) {
- if (AllRadioOptions[x].checked && ThisChecked == 'No') {
- ThisChecked = \"Yes\";
- break;
- }
- }
-
- var AlreadySearched = ShowAlert.indexOf(ThisRadio);
- if (ThisChecked == 'No' && AlreadySearched == -1) {
- ShowAlert = ShowAlert + ThisRadio;
- }
- }
- }
-
- if (ShowAlert != '') {
- $('.question-validate-btn').attr('disabled', 'disabled');
- } else {
- $('.question-validate-btn').removeAttr('disabled');
- }
- });
- </script>";
- foreach ($objQuestionTmp->optionsTitle as $item) {
- if (in_array($item, $objQuestionTmp->optionsTitle)) {
- $properties = [];
- if ($item === 'Answers') {
- $properties['colspan'] = 2;
- $properties['style'] = 'background-color: #F56B2A; color: #ffffff;';
- } elseif ($item == 'DegreeOfCertaintyThatMyAnswerIsCorrect') {
- $properties['colspan'] = 6;
- $properties['style'] = 'background-color: #330066; color: #ffffff;';
- }
- $header .= Display::tag('th', get_lang($item), $properties);
- } else {
- $header .= Display::tag('th', $item);
- }
- }
- if ($show_comment) {
- $header .= Display::tag('th', get_lang('Feedback'));
- }
- $s .= '<table class="data_table">';
- $s .= Display::tag('tr', $header, ['style' => 'text-align:left;']);
- // ajout de la 2eme ligne d'entête pour true/falss et les pourcentages de certitude
- $header1 = Display::tag('th', ' ');
- $cpt1 = 0;
- foreach ($objQuestionTmp->options as $item) {
- $colorBorder1 = ($cpt1 == (count($objQuestionTmp->options) - 1))
- ? '' : 'border-right: solid #FFFFFF 1px;';
- if ($item === 'True' || $item === 'False') {
- $header1 .= Display::tag(
- 'th',
- get_lang($item),
- ['style' => 'background-color: #F7C9B4; color: black;'.$colorBorder1]
- );
- } else {
- $header1 .= Display::tag(
- 'th',
- $item,
- ['style' => 'background-color: #e6e6ff; color: black;padding:5px; '.$colorBorder1]
- );
- }
- $cpt1++;
- }
- if ($show_comment) {
- $header1 .= Display::tag('th', ' ');
- }
- $s .= Display::tag('tr', $header1);
- // add explanation
- $header2 = Display::tag('th', ' ');
- $descriptionList = [
- get_lang('DegreeOfCertaintyIDeclareMyIgnorance'),
- get_lang('DegreeOfCertaintyIAmVeryUnsure'),
- get_lang('DegreeOfCertaintyIAmUnsure'),
- get_lang('DegreeOfCertaintyIAmPrettySure'),
- get_lang('DegreeOfCertaintyIAmSure'),
- get_lang('DegreeOfCertaintyIAmVerySure'),
- ];
- $counter2 = 0;
- foreach ($objQuestionTmp->options as $item) {
- if ($item == 'True' || $item == 'False') {
- $header2 .= Display::tag('td',
- ' ',
- ['style' => 'background-color: #F7E1D7; color: black;border-right: solid #FFFFFF 1px;']);
- } else {
- $color_border2 = ($counter2 == (count($objQuestionTmp->options) - 1)) ?
- '' : 'border-right: solid #FFFFFF 1px;font-size:11px;';
- $header2 .= Display::tag(
- 'td',
- nl2br($descriptionList[$counter2]),
- ['style' => 'background-color: #EFEFFC; color: black; width: 110px; text-align:center;
- vertical-align: top; padding:5px; '.$color_border2]);
- $counter2++;
- }
- }
- if ($show_comment) {
- $header2 .= Display::tag('th', ' ');
- }
- $s .= Display::tag('tr', $header2);
- }
- if ($show_comment) {
- if (in_array(
- $answerType,
- [
- MULTIPLE_ANSWER,
- MULTIPLE_ANSWER_COMBINATION,
- UNIQUE_ANSWER,
- UNIQUE_ANSWER_IMAGE,
- UNIQUE_ANSWER_NO_OPTION,
- GLOBAL_MULTIPLE_ANSWER,
- ]
- )) {
- $header = Display::tag('th', get_lang('Options'));
- if ($exercise->getFeedbackType() == EXERCISE_FEEDBACK_TYPE_END) {
- $header .= Display::tag('th', get_lang('Feedback'));
- }
- $s .= '<table class="table table-hover table-striped">';
- $s .= Display::tag(
- 'tr',
- $header,
- ['style' => 'text-align:left;']
- );
- }
- }
- $matching_correct_answer = 0;
- $userChoiceList = [];
- if (!empty($user_choice)) {
- foreach ($user_choice as $item) {
- $userChoiceList[] = $item['answer'];
- }
- }
- $hidingClass = '';
- if ($answerType == READING_COMPREHENSION) {
- $objQuestionTmp->setExerciseType($exercise->selectType());
- $objQuestionTmp->processText($objQuestionTmp->selectDescription());
- $hidingClass = 'hide-reading-answers';
- $s .= Display::div(
- $objQuestionTmp->selectTitle(),
- ['class' => 'question_title '.$hidingClass]
- );
- }
- for ($answerId = 1; $answerId <= $nbrAnswers; $answerId++) {
- $answer = $objAnswerTmp->selectAnswer($answerId);
- $answerCorrect = $objAnswerTmp->isCorrect($answerId);
- $numAnswer = $objAnswerTmp->selectAutoId($answerId);
- $comment = $objAnswerTmp->selectComment($answerId);
- $attributes = [];
- switch ($answerType) {
- case UNIQUE_ANSWER:
- case UNIQUE_ANSWER_NO_OPTION:
- case UNIQUE_ANSWER_IMAGE:
- case READING_COMPREHENSION:
- $input_id = 'choice-'.$questionId.'-'.$answerId;
- if (isset($user_choice[0]['answer']) && $user_choice[0]['answer'] == $numAnswer) {
- $attributes = [
- 'id' => $input_id,
- 'checked' => 1,
- 'selected' => 1,
- ];
- } else {
- $attributes = ['id' => $input_id];
- }
- if ($debug_mark_answer) {
- if ($answerCorrect) {
- $attributes['checked'] = 1;
- $attributes['selected'] = 1;
- }
- }
- if ($show_comment) {
- $s .= '<tr><td>';
- }
- if ($answerType == UNIQUE_ANSWER_IMAGE) {
- if ($show_comment) {
- if (empty($comment)) {
- $s .= '<div id="answer'.$questionId.$numAnswer.'"
- class="exercise-unique-answer-image" style="text-align: center">';
- } else {
- $s .= '<div id="answer'.$questionId.$numAnswer.'"
- class="exercise-unique-answer-image col-xs-6 col-sm-12"
- style="text-align: center">';
- }
- } else {
- $s .= '<div id="answer'.$questionId.$numAnswer.'"
- class="exercise-unique-answer-image col-xs-6 col-md-3"
- style="text-align: center">';
- }
- }
- $answer = Security::remove_XSS($answer, STUDENT);
- $s .= Display::input(
- 'hidden',
- 'choice2['.$questionId.']',
- '0'
- );
- $answer_input = null;
- $attributes['class'] = 'checkradios';
- if ($answerType == UNIQUE_ANSWER_IMAGE) {
- $attributes['class'] = '';
- $attributes['style'] = 'display: none;';
- $answer = '<div class="thumbnail">'.$answer.'</div>';
- }
- $answer_input .= '<label class="radio '.$hidingClass.'">';
- $answer_input .= Display::input(
- 'radio',
- 'choice['.$questionId.']',
- $numAnswer,
- $attributes
- );
- $answer_input .= $answer;
- $answer_input .= '</label>';
- if ($answerType == UNIQUE_ANSWER_IMAGE) {
- $answer_input .= "</div>";
- }
- if ($show_comment) {
- $s .= $answer_input;
- $s .= '</td>';
- $s .= '<td>';
- $s .= $comment;
- $s .= '</td>';
- $s .= '</tr>';
- } else {
- $s .= $answer_input;
- }
- break;
- case MULTIPLE_ANSWER:
- case MULTIPLE_ANSWER_TRUE_FALSE:
- case GLOBAL_MULTIPLE_ANSWER:
- case MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY:
- $input_id = 'choice-'.$questionId.'-'.$answerId;
- $answer = Security::remove_XSS($answer, STUDENT);
- if (in_array($numAnswer, $userChoiceList)) {
- $attributes = [
- 'id' => $input_id,
- 'checked' => 1,
- 'selected' => 1,
- ];
- } else {
- $attributes = ['id' => $input_id];
- }
- if ($debug_mark_answer) {
- if ($answerCorrect) {
- $attributes['checked'] = 1;
- $attributes['selected'] = 1;
- }
- }
- if ($answerType == MULTIPLE_ANSWER || $answerType == GLOBAL_MULTIPLE_ANSWER) {
- $s .= '<input type="hidden" name="choice2['.$questionId.']" value="0" />';
- $attributes['class'] = 'checkradios';
- $answer_input = '<label class="checkbox">';
- $answer_input .= Display::input(
- 'checkbox',
- 'choice['.$questionId.']['.$numAnswer.']',
- $numAnswer,
- $attributes
- );
- $answer_input .= $answer;
- $answer_input .= '</label>';
- if ($show_comment) {
- $s .= '<tr><td>';
- $s .= $answer_input;
- $s .= '</td>';
- $s .= '<td>';
- $s .= $comment;
- $s .= '</td>';
- $s .= '</tr>';
- } else {
- $s .= $answer_input;
- }
- } elseif ($answerType == MULTIPLE_ANSWER_TRUE_FALSE) {
- $myChoice = [];
- if (!empty($userChoiceList)) {
- foreach ($userChoiceList as $item) {
- $item = explode(':', $item);
- if (!empty($item)) {
- $myChoice[$item[0]] = isset($item[1]) ? $item[1] : '';
- }
- }
- }
- $s .= '<tr>';
- $s .= Display::tag('td', $answer);
- if (!empty($quizQuestionOptions)) {
- foreach ($quizQuestionOptions as $id => $item) {
- if (isset($myChoice[$numAnswer]) && $id == $myChoice[$numAnswer]) {
- $attributes = [
- 'checked' => 1,
- 'selected' => 1,
- ];
- } else {
- $attributes = [];
- }
- if ($debug_mark_answer) {
- if ($id == $answerCorrect) {
- $attributes['checked'] = 1;
- $attributes['selected'] = 1;
- }
- }
- $s .= Display::tag(
- 'td',
- Display::input(
- 'radio',
- 'choice['.$questionId.']['.$numAnswer.']',
- $id,
- $attributes
- ),
- ['style' => '']
- );
- }
- }
- if ($show_comment) {
- $s .= '<td>';
- $s .= $comment;
- $s .= '</td>';
- }
- $s .= '</tr>';
- } elseif ($answerType == MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY) {
- $myChoice = [];
- if (!empty($userChoiceList)) {
- foreach ($userChoiceList as $item) {
- $item = explode(':', $item);
- $myChoice[$item[0]] = $item[1];
- }
- }
- $myChoiceDegreeCertainty = [];
- if (!empty($userChoiceList)) {
- foreach ($userChoiceList as $item) {
- $item = explode(':', $item);
- $myChoiceDegreeCertainty[$item[0]] = $item[2];
- }
- }
- $s .= '<tr>';
- $s .= Display::tag('td', $answer);
- if (!empty($quizQuestionOptions)) {
- foreach ($quizQuestionOptions as $id => $item) {
- if (isset($myChoice[$numAnswer]) && $id == $myChoice[$numAnswer]) {
- $attributes = ['checked' => 1, 'selected' => 1];
- } else {
- $attributes = [];
- }
- $attributes['onChange'] = 'RadioValidator('.$questionId.', '.$numAnswer.')';
- // radio button selection
- if (isset($myChoiceDegreeCertainty[$numAnswer]) &&
- $id == $myChoiceDegreeCertainty[$numAnswer]
- ) {
- $attributes1 = ['checked' => 1, 'selected' => 1];
- } else {
- $attributes1 = [];
- }
- $attributes1['onChange'] = 'RadioValidator('.$questionId.', '.$numAnswer.')';
- if ($debug_mark_answer) {
- if ($id == $answerCorrect) {
- $attributes['checked'] = 1;
- $attributes['selected'] = 1;
- }
- }
- if ($item['name'] == 'True' || $item['name'] == 'False') {
- $s .= Display::tag('td',
- Display::input('radio',
- 'choice['.$questionId.']['.$numAnswer.']',
- $id,
- $attributes
- ),
- ['style' => 'text-align:center; background-color:#F7E1D7;',
- 'onclick' => 'handleRadioRow(event, '.
- $questionId.', '.
- $numAnswer.')',
- ]
- );
- } else {
- $s .= Display::tag('td',
- Display::input('radio',
- 'choiceDegreeCertainty['.$questionId.']['.$numAnswer.']',
- $id,
- $attributes1
- ),
- ['style' => 'text-align:center; background-color:#EFEFFC;',
- 'onclick' => 'handleRadioRow(event, '.
- $questionId.', '.
- $numAnswer.')',
- ]
- );
- }
- }
- }
- if ($show_comment) {
- $s .= '<td>';
- $s .= $comment;
- $s .= '</td>';
- }
- $s .= '</tr>';
- }
- break;
- case MULTIPLE_ANSWER_COMBINATION:
- // multiple answers
- $input_id = 'choice-'.$questionId.'-'.$answerId;
- if (in_array($numAnswer, $userChoiceList)) {
- $attributes = [
- 'id' => $input_id,
- 'checked' => 1,
- 'selected' => 1,
- ];
- } else {
- $attributes = ['id' => $input_id];
- }
- if ($debug_mark_answer) {
- if ($answerCorrect) {
- $attributes['checked'] = 1;
- $attributes['selected'] = 1;
- }
- }
- $answer = Security::remove_XSS($answer, STUDENT);
- $answer_input = '<input type="hidden" name="choice2['.$questionId.']" value="0" />';
- $answer_input .= '<label class="checkbox">';
- $answer_input .= Display::input(
- 'checkbox',
- 'choice['.$questionId.']['.$numAnswer.']',
- 1,
- $attributes
- );
- $answer_input .= $answer;
- $answer_input .= '</label>';
- if ($show_comment) {
- $s .= '<tr>';
- $s .= '<td>';
- $s .= $answer_input;
- $s .= '</td>';
- $s .= '<td>';
- $s .= $comment;
- $s .= '</td>';
- $s .= '</tr>';
- } else {
- $s .= $answer_input;
- }
- break;
- case MULTIPLE_ANSWER_COMBINATION_TRUE_FALSE:
- $s .= '<input type="hidden" name="choice2['.$questionId.']" value="0" />';
- $myChoice = [];
- if (!empty($userChoiceList)) {
- foreach ($userChoiceList as $item) {
- $item = explode(':', $item);
- if (isset($item[1]) && isset($item[0])) {
- $myChoice[$item[0]] = $item[1];
- }
- }
- }
- $answer = Security::remove_XSS($answer, STUDENT);
- $s .= '<tr>';
- $s .= Display::tag('td', $answer);
- foreach ($objQuestionTmp->options as $key => $item) {
- if (isset($myChoice[$numAnswer]) && $key == $myChoice[$numAnswer]) {
- $attributes = [
- 'checked' => 1,
- 'selected' => 1,
- ];
- } else {
- $attributes = [];
- }
- if ($debug_mark_answer) {
- if ($key == $answerCorrect) {
- $attributes['checked'] = 1;
- $attributes['selected'] = 1;
- }
- }
- $s .= Display::tag(
- 'td',
- Display::input(
- 'radio',
- 'choice['.$questionId.']['.$numAnswer.']',
- $key,
- $attributes
- )
- );
- }
- if ($show_comment) {
- $s .= '<td>';
- $s .= $comment;
- $s .= '</td>';
- }
- $s .= '</tr>';
- break;
- case FILL_IN_BLANKS:
- // display the question, with field empty, for student to fill it,
- // or filled to display the answer in the Question preview of the exercise/admin.php page
- $displayForStudent = true;
- $listAnswerInfo = FillBlanks::getAnswerInfo($answer);
- // Correct answers
- $correctAnswerList = $listAnswerInfo['words'];
- // Student's answer
- $studentAnswerList = [];
- if (isset($user_choice[0]['answer'])) {
- $arrayStudentAnswer = FillBlanks::getAnswerInfo(
- $user_choice[0]['answer'],
- true
- );
- $studentAnswerList = $arrayStudentAnswer['student_answer'];
- }
- // If the question must be shown with the answer (in page exercise/admin.php)
- // for teacher preview set the student-answer to the correct answer
- if ($debug_mark_answer) {
- $studentAnswerList = $correctAnswerList;
- $displayForStudent = false;
- }
- if (!empty($correctAnswerList) && !empty($studentAnswerList)) {
- $answer = '';
- for ($i = 0; $i < count($listAnswerInfo['common_words']) - 1; $i++) {
- // display the common word
- $answer .= $listAnswerInfo['common_words'][$i];
- // display the blank word
- $correctItem = $listAnswerInfo['words'][$i];
- if (isset($studentAnswerList[$i])) {
- // If student already started this test and answered this question,
- // fill the blank with his previous answers
- // may be "" if student viewed the question, but did not fill the blanks
- $correctItem = $studentAnswerList[$i];
- }
- $attributes['style'] = 'width:'.$listAnswerInfo['input_size'][$i].'px';
- $answer .= FillBlanks::getFillTheBlankHtml(
- $current_item,
- $questionId,
- $correctItem,
- $attributes,
- $answer,
- $listAnswerInfo,
- $displayForStudent,
- $i
- );
- }
- // display the last common word
- $answer .= $listAnswerInfo['common_words'][$i];
- } else {
- // display empty [input] with the right width for student to fill it
- $answer = '';
- for ($i = 0; $i < count($listAnswerInfo['common_words']) - 1; $i++) {
- // display the common words
- $answer .= $listAnswerInfo['common_words'][$i];
- // display the blank word
- $attributes['style'] = 'width:'.$listAnswerInfo['input_size'][$i].'px';
- $answer .= FillBlanks::getFillTheBlankHtml(
- $current_item,
- $questionId,
- '',
- $attributes,
- $answer,
- $listAnswerInfo,
- $displayForStudent,
- $i
- );
- }
- // display the last common word
- $answer .= $listAnswerInfo['common_words'][$i];
- }
- $s .= $answer;
- break;
- case CALCULATED_ANSWER:
- /*
- * In the CALCULATED_ANSWER test
- * you mustn't have [ and ] in the textarea
- * you mustn't have @@ in the textarea
- * the text to find mustn't be empty or contains only spaces
- * the text to find mustn't contains HTML tags
- * the text to find mustn't contains char "
- */
- if ($origin !== null) {
- global $exe_id;
- $exe_id = (int) $exe_id;
- $trackAttempts = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $sql = "SELECT answer FROM $trackAttempts
- WHERE exe_id = $exe_id AND question_id= $questionId";
- $rsLastAttempt = Database::query($sql);
- $rowLastAttempt = Database::fetch_array($rsLastAttempt);
- $answer = $rowLastAttempt['answer'];
- if (empty($answer)) {
- $_SESSION['calculatedAnswerId'][$questionId] = mt_rand(
- 1,
- $nbrAnswers
- );
- $answer = $objAnswerTmp->selectAnswer(
- $_SESSION['calculatedAnswerId'][$questionId]
- );
- }
- }
- list($answer) = explode('@@', $answer);
- // $correctAnswerList array of array with correct anwsers 0=> [0=>[\p] 1=>[plop]]
- api_preg_match_all(
- '/\[[^]]+\]/',
- $answer,
- $correctAnswerList
- );
- // get student answer to display it if student go back
- // to previous calculated answer question in a test
- if (isset($user_choice[0]['answer'])) {
- api_preg_match_all(
- '/\[[^]]+\]/',
- $answer,
- $studentAnswerList
- );
- $studentAnswerListToClean = $studentAnswerList[0];
- $studentAnswerList = [];
- $maxStudents = count($studentAnswerListToClean);
- for ($i = 0; $i < $maxStudents; $i++) {
- $answerCorrected = $studentAnswerListToClean[$i];
- $answerCorrected = api_preg_replace(
- '| / <font color="green"><b>.*$|',
- '',
- $answerCorrected
- );
- $answerCorrected = api_preg_replace(
- '/^\[/',
- '',
- $answerCorrected
- );
- $answerCorrected = api_preg_replace(
- '|^<font color="red"><s>|',
- '',
- $answerCorrected
- );
- $answerCorrected = api_preg_replace(
- '|</s></font>$|',
- '',
- $answerCorrected
- );
- $answerCorrected = '['.$answerCorrected.']';
- $studentAnswerList[] = $answerCorrected;
- }
- }
- // If display preview of answer in test view for exemple,
- // set the student answer to the correct answers
- if ($debug_mark_answer) {
- // contain the rights answers surronded with brackets
- $studentAnswerList = $correctAnswerList[0];
- }
- /*
- Split the response by bracket
- tabComments is an array with text surrounding the text to find
- we add a space before and after the answerQuestion to be sure to
- have a block of text before and after [xxx] patterns
- so we have n text to find ([xxx]) and n+1 block of texts before,
- between and after the text to find
- */
- $tabComments = api_preg_split(
- '/\[[^]]+\]/',
- ' '.$answer.' '
- );
- if (!empty($correctAnswerList) && !empty($studentAnswerList)) {
- $answer = '';
- $i = 0;
- foreach ($studentAnswerList as $studentItem) {
- // Remove surronding brackets
- $studentResponse = api_substr(
- $studentItem,
- 1,
- api_strlen($studentItem) - 2
- );
- $size = strlen($studentItem);
- $attributes['class'] = self::detectInputAppropriateClass(
- $size
- );
- $answer .= $tabComments[$i].
- Display::input(
- 'text',
- "choice[$questionId][]",
- $studentResponse,
- $attributes
- );
- $i++;
- }
- $answer .= $tabComments[$i];
- } else {
- // display exercise with empty input fields
- // every [xxx] are replaced with an empty input field
- foreach ($correctAnswerList[0] as $item) {
- $size = strlen($item);
- $attributes['class'] = self::detectInputAppropriateClass(
- $size
- );
- $answer = str_replace(
- $item,
- Display::input(
- 'text',
- "choice[$questionId][]",
- '',
- $attributes
- ),
- $answer
- );
- }
- }
- if ($origin !== null) {
- $s = $answer;
- break;
- } else {
- $s .= $answer;
- }
- break;
- case MATCHING:
- // matching type, showing suggestions and answers
- // TODO: replace $answerId by $numAnswer
- if ($answerCorrect != 0) {
- // only show elements to be answered (not the contents of
- // the select boxes, who are correct = 0)
- $s .= '<tr><td width="45%" valign="top">';
- $parsed_answer = $answer;
- // Left part questions
- $s .= '<p class="indent">'.$lines_count.'. '.$parsed_answer.'</p></td>';
- // Middle part (matches selects)
- // Id of select is # question + # of option
- $s .= '<td width="10%" valign="top" align="center">
- <div class="select-matching">
- <select
- id="choice_id_'.$current_item.'_'.$lines_count.'"
- name="choice['.$questionId.']['.$numAnswer.']">';
- // fills the list-box
- foreach ($select_items as $key => $val) {
- // set $debug_mark_answer to true at function start to
- // show the correct answer with a suffix '-x'
- $selected = '';
- if ($debug_mark_answer) {
- if ($val['id'] == $answerCorrect) {
- $selected = 'selected="selected"';
- }
- }
- //$user_choice_array_position
- if (isset($user_choice_array_position[$numAnswer]) &&
- $val['id'] == $user_choice_array_position[$numAnswer]
- ) {
- $selected = 'selected="selected"';
- }
- $s .= '<option value="'.$val['id'].'" '.$selected.'>'.$val['letter'].'</option>';
- }
- $s .= '</select></div></td><td width="5%" class="separate"> </td>';
- $s .= '<td width="40%" valign="top" >';
- if (isset($select_items[$lines_count])) {
- $s .= '<div class="text-right">
- <p class="indent">'.
- $select_items[$lines_count]['letter'].'. '.
- $select_items[$lines_count]['answer'].'
- </p>
- </div>';
- } else {
- $s .= ' ';
- }
- $s .= '</td>';
- $s .= '</tr>';
- $lines_count++;
- // If the left side of the "matching" has been completely
- // shown but the right side still has values to show...
- if (($lines_count - 1) == $num_suggestions) {
- // if it remains answers to shown at the right side
- while (isset($select_items[$lines_count])) {
- $s .= '<tr>
- <td colspan="2"></td>
- <td valign="top">';
- $s .= '<b>'.$select_items[$lines_count]['letter'].'.</b> '.
- $select_items[$lines_count]['answer'];
- $s .= "</td>
- </tr>";
- $lines_count++;
- }
- }
- $matching_correct_answer++;
- }
- break;
- case DRAGGABLE:
- if ($answerCorrect) {
- $windowId = $questionId.'_'.$lines_count;
- $s .= '<li class="touch-items" id="'.$windowId.'">';
- $s .= Display::div(
- $answer,
- [
- 'id' => "window_$windowId",
- 'class' => "window{$questionId}_question_draggable exercise-draggable-answer-option",
- ]
- );
- $draggableSelectOptions = [];
- $selectedValue = 0;
- $selectedIndex = 0;
- if ($user_choice) {
- foreach ($user_choice as $chosen) {
- if ($answerCorrect != $chosen['answer']) {
- continue;
- }
- $selectedValue = $chosen['answer'];
- }
- }
- foreach ($select_items as $key => $select_item) {
- $draggableSelectOptions[$select_item['id']] = $select_item['letter'];
- }
- foreach ($draggableSelectOptions as $value => $text) {
- if ($value == $selectedValue) {
- break;
- }
- $selectedIndex++;
- }
- $s .= Display::select(
- "choice[$questionId][$numAnswer]",
- $draggableSelectOptions,
- $selectedValue,
- [
- 'id' => "window_{$windowId}_select",
- 'class' => 'select_option hidden',
- ],
- false
- );
- if ($selectedValue && $selectedIndex) {
- $s .= "
- <script>
- $(function() {
- DraggableAnswer.deleteItem(
- $('#{$questionId}_$lines_count'),
- $('#drop_{$questionId}_{$selectedIndex}')
- );
- });
- </script>
- ";
- }
- if (isset($select_items[$lines_count])) {
- $s .= Display::div(
- Display::tag(
- 'b',
- $select_items[$lines_count]['letter']
- ).$select_items[$lines_count]['answer'],
- [
- 'id' => "window_{$windowId}_answer",
- 'class' => 'hidden',
- ]
- );
- } else {
- $s .= ' ';
- }
- $lines_count++;
- if (($lines_count - 1) == $num_suggestions) {
- while (isset($select_items[$lines_count])) {
- $s .= Display::tag('b', $select_items[$lines_count]['letter']);
- $s .= $select_items[$lines_count]['answer'];
- $lines_count++;
- }
- }
- $matching_correct_answer++;
- $s .= '</li>';
- }
- break;
- case MATCHING_DRAGGABLE:
- if ($answerId == 1) {
- echo $objAnswerTmp->getJs();
- }
- if ($answerCorrect != 0) {
- $windowId = "{$questionId}_{$lines_count}";
- $s .= <<<HTML
- <tr>
- <td width="45%">
- <div id="window_{$windowId}"
- class="window window_left_question window{$questionId}_question">
- <strong>$lines_count.</strong>
- $answer
- </div>
- </td>
- <td width="10%">
- HTML;
- $draggableSelectOptions = [];
- $selectedValue = 0;
- $selectedIndex = 0;
- if ($user_choice) {
- foreach ($user_choice as $chosen) {
- if ($numAnswer == $chosen['position']) {
- $selectedValue = $chosen['answer'];
- break;
- }
- }
- }
- foreach ($select_items as $key => $selectItem) {
- $draggableSelectOptions[$selectItem['id']] = $selectItem['letter'];
- }
- foreach ($draggableSelectOptions as $value => $text) {
- if ($value == $selectedValue) {
- break;
- }
- $selectedIndex++;
- }
- $s .= Display::select(
- "choice[$questionId][$numAnswer]",
- $draggableSelectOptions,
- $selectedValue,
- [
- 'id' => "window_{$windowId}_select",
- 'class' => 'hidden',
- ],
- false
- );
- if (!empty($answerCorrect) && !empty($selectedValue)) {
- // Show connect if is not freeze (question preview)
- if (!$freeze) {
- $s .= "
- <script>
- $(function() {
- jsPlumb.ready(function() {
- jsPlumb.connect({
- source: 'window_$windowId',
- target: 'window_{$questionId}_{$selectedIndex}_answer',
- endpoint: ['Blank', {radius: 15}],
- anchors: ['RightMiddle', 'LeftMiddle'],
- paintStyle: {strokeStyle: '#8A8888', lineWidth: 8},
- connector: [
- MatchingDraggable.connectorType,
- {curvines: MatchingDraggable.curviness}
- ]
- });
- });
- });
- </script>
- ";
- }
- }
- $s .= '</td><td width="45%">';
- if (isset($select_items[$lines_count])) {
- $s .= <<<HTML
- <div id="window_{$windowId}_answer" class="window window_right_question">
- <strong>{$select_items[$lines_count]['letter']}.</strong>
- {$select_items[$lines_count]['answer']}
- </div>
- HTML;
- } else {
- $s .= ' ';
- }
- $s .= '</td></tr>';
- $lines_count++;
- if (($lines_count - 1) == $num_suggestions) {
- while (isset($select_items[$lines_count])) {
- $s .= <<<HTML
- <tr>
- <td colspan="2"></td>
- <td>
- <strong>{$select_items[$lines_count]['letter']}</strong>
- {$select_items[$lines_count]['answer']}
- </td>
- </tr>
- HTML;
- $lines_count++;
- }
- }
- $matching_correct_answer++;
- }
- break;
- }
- }
- if ($show_comment) {
- $s .= '</table>';
- } elseif (in_array(
- $answerType,
- [
- MATCHING,
- MATCHING_DRAGGABLE,
- UNIQUE_ANSWER_NO_OPTION,
- MULTIPLE_ANSWER_TRUE_FALSE,
- MULTIPLE_ANSWER_COMBINATION_TRUE_FALSE,
- MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY,
- ]
- )) {
- $s .= '</table>';
- }
- if ($answerType == DRAGGABLE) {
- $isVertical = $objQuestionTmp->extra == 'v';
- $s .= "</ul>";
- $s .= "</div>";
- $counterAnswer = 1;
- $s .= $isVertical ? '' : '<div class="row">';
- for ($answerId = 1; $answerId <= $nbrAnswers; $answerId++) {
- $answerCorrect = $objAnswerTmp->isCorrect($answerId);
- $windowId = $questionId.'_'.$counterAnswer;
- if ($answerCorrect) {
- $s .= $isVertical ? '<div class="row">' : '';
- $s .= '
- <div class="'.($isVertical ? 'col-md-12' : 'col-xs-12 col-sm-4 col-md-3 col-lg-2').'">
- <div class="droppable-item">
- <span class="number">'.$counterAnswer.'.</span>
- <div id="drop_'.$windowId.'" class="droppable"> </div>
- </div>
- </div>
- ';
- $s .= $isVertical ? '</div>' : '';
- $counterAnswer++;
- }
- }
- $s .= $isVertical ? '' : '</div>'; // row
- $s .= '</div>'; // col-md-12 ui-widget ui-helper-clearfix
- }
- if (in_array($answerType, [MATCHING, MATCHING_DRAGGABLE])) {
- $s .= '</div>'; //drag_question
- }
- $s .= '</div>'; //question_options row
- // destruction of the Answer object
- unset($objAnswerTmp);
- // destruction of the Question object
- unset($objQuestionTmp);
- if ($origin == 'export') {
- return $s;
- }
- echo $s;
- } elseif ($answerType == HOT_SPOT || $answerType == HOT_SPOT_DELINEATION) {
- global $exe_id;
- // Question is a HOT_SPOT
- // Checking document/images visibility
- if (api_is_platform_admin() || api_is_course_admin()) {
- $doc_id = $objQuestionTmp->getPictureId();
- if (is_numeric($doc_id)) {
- $images_folder_visibility = api_get_item_visibility(
- $course,
- 'document',
- $doc_id,
- api_get_session_id()
- );
- if (!$images_folder_visibility) {
- // Show only to the course/platform admin if the image is set to visibility = false
- echo Display::return_message(
- get_lang('ChangeTheVisibilityOfTheCurrentImage'),
- 'warning'
- );
- }
- }
- }
- $questionDescription = $objQuestionTmp->selectDescription();
- // Get the answers, make a list
- $objAnswerTmp = new Answer($questionId, $course_id);
- $nbrAnswers = $objAnswerTmp->selectNbrAnswers();
- // get answers of hotpost
- $answers_hotspot = [];
- for ($answerId = 1; $answerId <= $nbrAnswers; $answerId++) {
- $answers = $objAnswerTmp->selectAnswerByAutoId(
- $objAnswerTmp->selectAutoId($answerId)
- );
- $answers_hotspot[$answers['id']] = $objAnswerTmp->selectAnswer(
- $answerId
- );
- }
- $answerList = '';
- $hotspotColor = 0;
- if ($answerType != HOT_SPOT_DELINEATION) {
- $answerList = '
- <div class="well well-sm">
- <h5 class="page-header">'.get_lang('HotspotZones').'</h5>
- <ol>
- ';
- if (!empty($answers_hotspot)) {
- Session::write("hotspot_ordered$questionId", array_keys($answers_hotspot));
- foreach ($answers_hotspot as $value) {
- $answerList .= '<li>';
- if ($freeze) {
- $answerList .= '<span class="hotspot-color-'.$hotspotColor
- .' fa fa-square" aria-hidden="true"></span>'.PHP_EOL;
- }
- $answerList .= $value;
- $answerList .= '</li>';
- $hotspotColor++;
- }
- }
- $answerList .= '
- </ul>
- </div>
- ';
- if ($freeze) {
- $relPath = api_get_path(WEB_CODE_PATH);
- echo "
- <div class=\"row\">
- <div class=\"col-sm-9\">
- <div id=\"hotspot-preview-$questionId\"></div>
- </div>
- <div class=\"col-sm-3\">
- $answerList
- </div>
- </div>
- <script>
- new ".($answerType == HOT_SPOT ? "HotspotQuestion" : "DelineationQuestion")."({
- questionId: $questionId,
- exerciseId: $exerciseId,
- exeId: 0,
- selector: '#hotspot-preview-$questionId',
- for: 'preview',
- relPath: '$relPath'
- });
- </script>
- ";
- return;
- }
- }
- if (!$only_questions) {
- if ($show_title) {
- if ($exercise->display_category_name) {
- TestCategory::displayCategoryAndTitle($objQuestionTmp->id);
- }
- echo $objQuestionTmp->getTitleToDisplay($current_item);
- }
- //@todo I need to the get the feedback type
- echo <<<HOTSPOT
- <input type="hidden" name="hidden_hotspot_id" value="$questionId" />
- <div class="exercise_questions">
- $questionDescription
- <div class="row">
- HOTSPOT;
- }
- $relPath = api_get_path(WEB_CODE_PATH);
- $s .= "<div class=\"col-sm-8 col-md-9\">
- <div class=\"hotspot-image\"></div>
- <script>
- $(function() {
- new ".($answerType == HOT_SPOT_DELINEATION ? 'DelineationQuestion' : 'HotspotQuestion')."({
- questionId: $questionId,
- exerciseId: $exerciseId,
- selector: '#question_div_' + $questionId + ' .hotspot-image',
- for: 'user',
- relPath: '$relPath'
- });
- });
- </script>
- </div>
- <div class=\"col-sm-4 col-md-3\">
- $answerList
- </div>
- ";
- echo <<<HOTSPOT
- $s
- </div>
- </div>
- HOTSPOT;
- } elseif ($answerType == ANNOTATION) {
- global $exe_id;
- $relPath = api_get_path(WEB_CODE_PATH);
- if (api_is_platform_admin() || api_is_course_admin()) {
- $docId = DocumentManager::get_document_id($course, '/images/'.$pictureName);
- if ($docId) {
- $images_folder_visibility = api_get_item_visibility(
- $course,
- 'document',
- $docId,
- api_get_session_id()
- );
- if (!$images_folder_visibility) {
- echo Display::return_message(get_lang('ChangeTheVisibilityOfTheCurrentImage'), 'warning');
- }
- }
- if ($freeze) {
- echo Display::img(
- api_get_path(WEB_COURSE_PATH).$course['path'].'/document/images/'.$pictureName,
- $objQuestionTmp->selectTitle(),
- ['width' => '600px']
- );
- return 0;
- }
- }
- if (!$only_questions) {
- if ($show_title) {
- if ($exercise->display_category_name) {
- TestCategory::displayCategoryAndTitle($objQuestionTmp->id);
- }
- echo $objQuestionTmp->getTitleToDisplay($current_item);
- }
- echo '
- <input type="hidden" name="hidden_hotspot_id" value="'.$questionId.'" />
- <div class="exercise_questions">
- '.$objQuestionTmp->selectDescription().'
- <div class="row">
- <div class="col-sm-8 col-md-9">
- <div id="annotation-canvas-'.$questionId.'" class="annotation-canvas center-block">
- </div>
- <script>
- AnnotationQuestion({
- questionId: '.$questionId.',
- exerciseId: '.$exerciseId.',
- relPath: \''.$relPath.'\',
- courseId: '.$course_id.',
- });
- </script>
- </div>
- <div class="col-sm-4 col-md-3">
- <div class="well well-sm" id="annotation-toolbar-'.$questionId.'">
- <div class="btn-toolbar">
- <div class="btn-group" data-toggle="buttons">
- <label class="btn btn-default active"
- aria-label="'.get_lang('AddAnnotationPath').'">
- <input
- type="radio" value="0"
- name="'.$questionId.'-options" autocomplete="off" checked>
- <span class="fa fa-pencil" aria-hidden="true"></span>
- </label>
- <label class="btn btn-default"
- aria-label="'.get_lang('AddAnnotationText').'">
- <input
- type="radio" value="1"
- name="'.$questionId.'-options" autocomplete="off">
- <span class="fa fa-font fa-fw" aria-hidden="true"></span>
- </label>
- </div>
- </div>
- <ul class="list-unstyled"></ul>
- </div>
- </div>
- </div>
- </div>
- ';
- }
- $objAnswerTmp = new Answer($questionId);
- $nbrAnswers = $objAnswerTmp->selectNbrAnswers();
- unset($objAnswerTmp, $objQuestionTmp);
- }
- return $nbrAnswers;
- }
- /**
- * @param int $exeId
- *
- * @return array
- */
- public static function get_exercise_track_exercise_info($exeId)
- {
- $quizTable = Database::get_course_table(TABLE_QUIZ_TEST);
- $trackExerciseTable = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $courseTable = Database::get_main_table(TABLE_MAIN_COURSE);
- $exeId = (int) $exeId;
- $result = [];
- if (!empty($exeId)) {
- $sql = " SELECT q.*, tee.*
- FROM $quizTable as q
- INNER JOIN $trackExerciseTable as tee
- ON q.id = tee.exe_exo_id
- INNER JOIN $courseTable c
- ON c.id = tee.c_id
- WHERE tee.exe_id = $exeId
- AND q.c_id = c.id";
- $sqlResult = Database::query($sql);
- if (Database::num_rows($sqlResult)) {
- $result = Database::fetch_array($sqlResult, 'ASSOC');
- $result['duration_formatted'] = '';
- if (!empty($result['exe_duration'])) {
- $time = api_format_time($result['exe_duration'], 'js');
- $result['duration_formatted'] = $time;
- }
- }
- }
- return $result;
- }
- /**
- * Validates the time control key.
- *
- * @param int $exercise_id
- * @param int $lp_id
- * @param int $lp_item_id
- *
- * @return bool
- */
- public static function exercise_time_control_is_valid(
- $exercise_id,
- $lp_id = 0,
- $lp_item_id = 0
- ) {
- $course_id = api_get_course_int_id();
- $exercise_id = (int) $exercise_id;
- $table = Database::get_course_table(TABLE_QUIZ_TEST);
- $sql = "SELECT expired_time FROM $table
- WHERE c_id = $course_id AND id = $exercise_id";
- $result = Database::query($sql);
- $row = Database::fetch_array($result, 'ASSOC');
- if (!empty($row['expired_time'])) {
- $current_expired_time_key = self::get_time_control_key(
- $exercise_id,
- $lp_id,
- $lp_item_id
- );
- if (isset($_SESSION['expired_time'][$current_expired_time_key])) {
- $current_time = time();
- $expired_time = api_strtotime(
- $_SESSION['expired_time'][$current_expired_time_key],
- 'UTC'
- );
- $total_time_allowed = $expired_time + 30;
- if ($total_time_allowed < $current_time) {
- return false;
- }
- return true;
- }
- return false;
- }
- return true;
- }
- /**
- * Deletes the time control token.
- *
- * @param int $exercise_id
- * @param int $lp_id
- * @param int $lp_item_id
- */
- public static function exercise_time_control_delete(
- $exercise_id,
- $lp_id = 0,
- $lp_item_id = 0
- ) {
- $current_expired_time_key = self::get_time_control_key(
- $exercise_id,
- $lp_id,
- $lp_item_id
- );
- unset($_SESSION['expired_time'][$current_expired_time_key]);
- }
- /**
- * Generates the time control key.
- *
- * @param int $exercise_id
- * @param int $lp_id
- * @param int $lp_item_id
- *
- * @return string
- */
- public static function get_time_control_key(
- $exercise_id,
- $lp_id = 0,
- $lp_item_id = 0
- ) {
- $exercise_id = (int) $exercise_id;
- $lp_id = (int) $lp_id;
- $lp_item_id = (int) $lp_item_id;
- return
- api_get_course_int_id().'_'.
- api_get_session_id().'_'.
- $exercise_id.'_'.
- api_get_user_id().'_'.
- $lp_id.'_'.
- $lp_item_id;
- }
- /**
- * Get session time control.
- *
- * @param int $exercise_id
- * @param int $lp_id
- * @param int $lp_item_id
- *
- * @return int
- */
- public static function get_session_time_control_key(
- $exercise_id,
- $lp_id = 0,
- $lp_item_id = 0
- ) {
- $return_value = 0;
- $time_control_key = self::get_time_control_key(
- $exercise_id,
- $lp_id,
- $lp_item_id
- );
- if (isset($_SESSION['expired_time']) && isset($_SESSION['expired_time'][$time_control_key])) {
- $return_value = $_SESSION['expired_time'][$time_control_key];
- }
- return $return_value;
- }
- /**
- * Gets count of exam results.
- *
- * @param int $exerciseId
- * @param array $conditions
- * @param string $courseCode
- * @param bool $showSession
- *
- * @return array
- */
- public static function get_count_exam_results($exerciseId, $conditions, $courseCode = '', $showSession = false)
- {
- $count = self::get_exam_results_data(
- null,
- null,
- null,
- null,
- $exerciseId,
- $conditions,
- true,
- $courseCode,
- $showSession
- );
- return $count;
- }
- /**
- * @param string $path
- *
- * @return int
- */
- public static function get_count_exam_hotpotatoes_results($path)
- {
- return self::get_exam_results_hotpotatoes_data(
- 0,
- 0,
- '',
- '',
- $path,
- true,
- ''
- );
- }
- /**
- * @param int $in_from
- * @param int $in_number_of_items
- * @param int $in_column
- * @param int $in_direction
- * @param string $in_hotpot_path
- * @param bool $in_get_count
- * @param null $where_condition
- *
- * @return array|int
- */
- public static function get_exam_results_hotpotatoes_data(
- $in_from,
- $in_number_of_items,
- $in_column,
- $in_direction,
- $in_hotpot_path,
- $in_get_count = false,
- $where_condition = null
- ) {
- $courseId = api_get_course_int_id();
- // by default in_column = 1 If parameters given, it is the name of the column witch is the bdd field name
- if ($in_column == 1) {
- $in_column = 'firstname';
- }
- $in_hotpot_path = Database::escape_string($in_hotpot_path);
- $in_direction = Database::escape_string($in_direction);
- $in_column = Database::escape_string($in_column);
- $in_number_of_items = intval($in_number_of_items);
- $in_from = (int) $in_from;
- $TBL_TRACK_HOTPOTATOES = Database::get_main_table(
- TABLE_STATISTIC_TRACK_E_HOTPOTATOES
- );
- $TBL_USER = Database::get_main_table(TABLE_MAIN_USER);
- $sql = "SELECT *, thp.id AS thp_id
- FROM $TBL_TRACK_HOTPOTATOES thp
- JOIN $TBL_USER u
- ON thp.exe_user_id = u.user_id
- WHERE
- thp.c_id = $courseId AND
- exe_name LIKE '$in_hotpot_path%'";
- // just count how many answers
- if ($in_get_count) {
- $res = Database::query($sql);
- return Database::num_rows($res);
- }
- // get a number of sorted results
- $sql .= " $where_condition
- ORDER BY $in_column $in_direction
- LIMIT $in_from, $in_number_of_items";
- $res = Database::query($sql);
- $result = [];
- $apiIsAllowedToEdit = api_is_allowed_to_edit();
- $urlBase = api_get_path(WEB_CODE_PATH).
- 'exercise/hotpotatoes_exercise_report.php?action=delete&'.
- api_get_cidreq().'&id=';
- while ($data = Database::fetch_array($res)) {
- $actions = null;
- if ($apiIsAllowedToEdit) {
- $url = $urlBase.$data['thp_id'].'&path='.$data['exe_name'];
- $actions = Display::url(
- Display::return_icon('delete.png', get_lang('Delete')),
- $url
- );
- }
- $result[] = [
- 'firstname' => $data['firstname'],
- 'lastname' => $data['lastname'],
- 'username' => $data['username'],
- 'group_name' => implode(
- '<br/>',
- GroupManager::get_user_group_name($data['user_id'])
- ),
- 'exe_date' => $data['exe_date'],
- 'score' => $data['exe_result'].' / '.$data['exe_weighting'],
- 'actions' => $actions,
- ];
- }
- return $result;
- }
- /**
- * @param string $exercisePath
- * @param int $userId
- * @param int $courseId
- * @param int $sessionId
- *
- * @return array
- */
- public static function getLatestHotPotatoResult(
- $exercisePath,
- $userId,
- $courseId,
- $sessionId
- ) {
- $table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_HOTPOTATOES);
- $exercisePath = Database::escape_string($exercisePath);
- $userId = (int) $userId;
- $courseId = (int) $courseId;
- $sql = "SELECT * FROM $table
- WHERE
- c_id = $courseId AND
- exe_name LIKE '$exercisePath%' AND
- exe_user_id = $userId
- ORDER BY id
- LIMIT 1";
- $result = Database::query($sql);
- $attempt = [];
- if (Database::num_rows($result)) {
- $attempt = Database::fetch_array($result, 'ASSOC');
- }
- return $attempt;
- }
- /**
- * Gets the exam'data results.
- *
- * @todo this function should be moved in a library + no global calls
- *
- * @param int $from
- * @param int $number_of_items
- * @param int $column
- * @param string $direction
- * @param int $exercise_id
- * @param null $extra_where_conditions
- * @param bool $get_count
- * @param string $courseCode
- * @param bool $showSessionField
- * @param bool $showExerciseCategories
- * @param array $userExtraFieldsToAdd
- * @param bool $useCommaAsDecimalPoint
- * @param bool $roundValues
- *
- * @return array
- */
- public static function get_exam_results_data(
- $from,
- $number_of_items,
- $column,
- $direction,
- $exercise_id,
- $extra_where_conditions = null,
- $get_count = false,
- $courseCode = null,
- $showSessionField = false,
- $showExerciseCategories = false,
- $userExtraFieldsToAdd = [],
- $useCommaAsDecimalPoint = false,
- $roundValues = false
- ) {
- //@todo replace all this globals
- global $filter;
- $courseCode = empty($courseCode) ? api_get_course_id() : $courseCode;
- $courseInfo = api_get_course_info($courseCode);
- if (empty($courseInfo)) {
- return [];
- }
- $documentPath = api_get_path(SYS_COURSE_PATH).$courseInfo['path'].'/document';
- $course_id = $courseInfo['real_id'];
- $sessionId = api_get_session_id();
- $exercise_id = (int) $exercise_id;
- $is_allowedToEdit =
- api_is_allowed_to_edit(null, true) ||
- api_is_allowed_to_edit(true) ||
- api_is_drh() ||
- api_is_student_boss() ||
- api_is_session_admin();
- $TBL_USER = Database::get_main_table(TABLE_MAIN_USER);
- $TBL_EXERCICES = Database::get_course_table(TABLE_QUIZ_TEST);
- $TBL_GROUP_REL_USER = Database::get_course_table(TABLE_GROUP_USER);
- $TBL_GROUP = Database::get_course_table(TABLE_GROUP);
- $TBL_TRACK_EXERCICES = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $TBL_TRACK_HOTPOTATOES = Database::get_main_table(TABLE_STATISTIC_TRACK_E_HOTPOTATOES);
- $TBL_TRACK_ATTEMPT_RECORDING = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT_RECORDING);
- $session_id_and = '';
- $sessionCondition = '';
- if (!$showSessionField) {
- $session_id_and = " AND te.session_id = $sessionId ";
- $sessionCondition = " AND ttte.session_id = $sessionId";
- }
- $exercise_where = '';
- if (!empty($exercise_id)) {
- $exercise_where .= ' AND te.exe_exo_id = '.$exercise_id.' ';
- }
- $hotpotatoe_where = '';
- if (!empty($_GET['path'])) {
- $hotpotatoe_path = Database::escape_string($_GET['path']);
- $hotpotatoe_where .= ' AND exe_name = "'.$hotpotatoe_path.'" ';
- }
- // sql for chamilo-type tests for teacher / tutor view
- $sql_inner_join_tbl_track_exercices = "
- (
- SELECT DISTINCT ttte.*, if(tr.exe_id,1, 0) as revised
- FROM $TBL_TRACK_EXERCICES ttte
- LEFT JOIN $TBL_TRACK_ATTEMPT_RECORDING tr
- ON (ttte.exe_id = tr.exe_id)
- WHERE
- c_id = $course_id AND
- exe_exo_id = $exercise_id
- $sessionCondition
- )";
- if ($is_allowedToEdit) {
- //@todo fix to work with COURSE_RELATION_TYPE_RRHH in both queries
- // Hack in order to filter groups
- $sql_inner_join_tbl_user = '';
- if (strpos($extra_where_conditions, 'group_id')) {
- $sql_inner_join_tbl_user = "
- (
- SELECT
- u.user_id,
- firstname,
- lastname,
- official_code,
- email,
- username,
- g.name as group_name,
- g.id as group_id
- FROM $TBL_USER u
- INNER JOIN $TBL_GROUP_REL_USER gru
- ON (gru.user_id = u.user_id AND gru.c_id= $course_id )
- INNER JOIN $TBL_GROUP g
- ON (gru.group_id = g.id AND g.c_id= $course_id )
- )";
- }
- if (strpos($extra_where_conditions, 'group_all')) {
- $extra_where_conditions = str_replace(
- "AND ( group_id = 'group_all' )",
- '',
- $extra_where_conditions
- );
- $extra_where_conditions = str_replace(
- "AND group_id = 'group_all'",
- '',
- $extra_where_conditions
- );
- $extra_where_conditions = str_replace(
- "group_id = 'group_all' AND",
- '',
- $extra_where_conditions
- );
- $sql_inner_join_tbl_user = "
- (
- SELECT
- u.user_id,
- firstname,
- lastname,
- official_code,
- email,
- username,
- '' as group_name,
- '' as group_id
- FROM $TBL_USER u
- )";
- $sql_inner_join_tbl_user = null;
- }
- if (strpos($extra_where_conditions, 'group_none')) {
- $extra_where_conditions = str_replace(
- "AND ( group_id = 'group_none' )",
- "AND ( group_id is null )",
- $extra_where_conditions
- );
- $extra_where_conditions = str_replace(
- "AND group_id = 'group_none'",
- "AND ( group_id is null )",
- $extra_where_conditions
- );
- $sql_inner_join_tbl_user = "
- (
- SELECT
- u.user_id,
- firstname,
- lastname,
- official_code,
- email,
- username,
- g.name as group_name,
- g.id as group_id
- FROM $TBL_USER u
- LEFT OUTER JOIN $TBL_GROUP_REL_USER gru
- ON ( gru.user_id = u.user_id AND gru.c_id= $course_id )
- LEFT OUTER JOIN $TBL_GROUP g
- ON (gru.group_id = g.id AND g.c_id = $course_id )
- )";
- }
- // All
- $is_empty_sql_inner_join_tbl_user = false;
- if (empty($sql_inner_join_tbl_user)) {
- $is_empty_sql_inner_join_tbl_user = true;
- $sql_inner_join_tbl_user = "
- (
- SELECT u.user_id, firstname, lastname, email, username, ' ' as group_name, '' as group_id, official_code
- FROM $TBL_USER u
- WHERE u.status NOT IN(".api_get_users_status_ignored_in_reports('string').")
- )";
- }
- $sqlFromOption = " , $TBL_GROUP_REL_USER AS gru ";
- $sqlWhereOption = " AND gru.c_id = $course_id AND gru.user_id = user.user_id ";
- $first_and_last_name = api_is_western_name_order() ? "firstname, lastname" : "lastname, firstname";
- if ($get_count) {
- $sql_select = 'SELECT count(te.exe_id) ';
- } else {
- $sql_select = "SELECT DISTINCT
- user_id,
- $first_and_last_name,
- official_code,
- ce.title,
- username,
- te.exe_result,
- te.exe_weighting,
- te.exe_date,
- te.exe_id,
- te.session_id,
- email as exemail,
- te.start_date,
- ce.expired_time,
- steps_counter,
- exe_user_id,
- te.exe_duration,
- te.status as completion_status,
- propagate_neg,
- revised,
- group_name,
- group_id,
- orig_lp_id,
- te.user_ip";
- }
- $sql = " $sql_select
- FROM $TBL_EXERCICES AS ce
- INNER JOIN $sql_inner_join_tbl_track_exercices AS te
- ON (te.exe_exo_id = ce.id)
- INNER JOIN $sql_inner_join_tbl_user AS user
- ON (user.user_id = exe_user_id)
- WHERE
- te.c_id = $course_id $session_id_and AND
- ce.active <> -1 AND
- ce.c_id = $course_id
- $exercise_where
- $extra_where_conditions
- ";
- // sql for hotpotatoes tests for teacher / tutor view
- if ($get_count) {
- $hpsql_select = ' SELECT count(username) ';
- } else {
- $hpsql_select = " SELECT
- $first_and_last_name ,
- username,
- official_code,
- tth.exe_name,
- tth.exe_result ,
- tth.exe_weighting,
- tth.exe_date";
- }
- $hpsql = " $hpsql_select
- FROM
- $TBL_TRACK_HOTPOTATOES tth,
- $TBL_USER user
- $sqlFromOption
- WHERE
- user.user_id=tth.exe_user_id
- AND tth.c_id = $course_id
- $hotpotatoe_where
- $sqlWhereOption
- AND user.status NOT IN (".api_get_users_status_ignored_in_reports('string').")
- ORDER BY tth.c_id ASC, tth.exe_date DESC ";
- }
- if (empty($sql)) {
- return false;
- }
- if ($get_count) {
- $resx = Database::query($sql);
- $rowx = Database::fetch_row($resx, 'ASSOC');
- return $rowx[0];
- }
- $teacher_list = CourseManager::get_teacher_list_from_course_code($courseCode);
- $teacher_id_list = [];
- if (!empty($teacher_list)) {
- foreach ($teacher_list as $teacher) {
- $teacher_id_list[] = $teacher['user_id'];
- }
- }
- $scoreDisplay = new ScoreDisplay();
- $decimalSeparator = '.';
- $thousandSeparator = ',';
- if ($useCommaAsDecimalPoint) {
- $decimalSeparator = ',';
- $thousandSeparator = '';
- }
- $listInfo = [];
- // Simple exercises
- if (empty($hotpotatoe_where)) {
- $column = !empty($column) ? Database::escape_string($column) : null;
- $from = (int) $from;
- $number_of_items = (int) $number_of_items;
- if (!empty($column)) {
- $sql .= " ORDER BY $column $direction ";
- }
- $sql .= " LIMIT $from, $number_of_items";
- $results = [];
- $resx = Database::query($sql);
- while ($rowx = Database::fetch_array($resx, 'ASSOC')) {
- $results[] = $rowx;
- }
- $group_list = GroupManager::get_group_list(null, $courseInfo);
- $clean_group_list = [];
- if (!empty($group_list)) {
- foreach ($group_list as $group) {
- $clean_group_list[$group['id']] = $group['name'];
- }
- }
- $lp_list_obj = new LearnpathList(api_get_user_id());
- $lp_list = $lp_list_obj->get_flat_list();
- $oldIds = array_column($lp_list, 'lp_old_id', 'iid');
- if (is_array($results)) {
- $users_array_id = [];
- $from_gradebook = false;
- if (isset($_GET['gradebook']) && $_GET['gradebook'] == 'view') {
- $from_gradebook = true;
- }
- $sizeof = count($results);
- $locked = api_resource_is_locked_by_gradebook(
- $exercise_id,
- LINK_EXERCISE
- );
- $timeNow = strtotime(api_get_utc_datetime());
- // Looping results
- for ($i = 0; $i < $sizeof; $i++) {
- $revised = $results[$i]['revised'];
- if ($results[$i]['completion_status'] == 'incomplete') {
- // If the exercise was incomplete, we need to determine
- // if it is still into the time allowed, or if its
- // allowed time has expired and it can be closed
- // (it's "unclosed")
- $minutes = $results[$i]['expired_time'];
- if ($minutes == 0) {
- // There's no time limit, so obviously the attempt
- // can still be "ongoing", but the teacher should
- // be able to choose to close it, so mark it as
- // "unclosed" instead of "ongoing"
- $revised = 2;
- } else {
- $allowedSeconds = $minutes * 60;
- $timeAttemptStarted = strtotime($results[$i]['start_date']);
- $secondsSinceStart = $timeNow - $timeAttemptStarted;
- if ($secondsSinceStart > $allowedSeconds) {
- $revised = 2; // mark as "unclosed"
- } else {
- $revised = 3; // mark as "ongoing"
- }
- }
- }
- if ($from_gradebook && ($is_allowedToEdit)) {
- if (in_array(
- $results[$i]['username'].$results[$i]['firstname'].$results[$i]['lastname'],
- $users_array_id
- )) {
- continue;
- }
- $users_array_id[] = $results[$i]['username'].$results[$i]['firstname'].$results[$i]['lastname'];
- }
- $lp_obj = isset($results[$i]['orig_lp_id']) && isset($lp_list[$results[$i]['orig_lp_id']]) ? $lp_list[$results[$i]['orig_lp_id']] : null;
- if (empty($lp_obj)) {
- // Try to get the old id (id instead of iid)
- $lpNewId = isset($results[$i]['orig_lp_id']) && isset($oldIds[$results[$i]['orig_lp_id']]) ? $oldIds[$results[$i]['orig_lp_id']] : null;
- if ($lpNewId) {
- $lp_obj = isset($lp_list[$lpNewId]) ? $lp_list[$lpNewId] : null;
- }
- }
- $lp_name = null;
- if ($lp_obj) {
- $url = api_get_path(WEB_CODE_PATH).'lp/lp_controller.php?'.api_get_cidreq().'&action=view&lp_id='.$results[$i]['orig_lp_id'];
- $lp_name = Display::url(
- $lp_obj['lp_name'],
- $url,
- ['target' => '_blank']
- );
- }
- // Add all groups by user
- $group_name_list = '';
- if ($is_empty_sql_inner_join_tbl_user) {
- $group_list = GroupManager::get_group_ids(
- api_get_course_int_id(),
- $results[$i]['user_id']
- );
- foreach ($group_list as $id) {
- if (isset($clean_group_list[$id])) {
- $group_name_list .= $clean_group_list[$id].'<br/>';
- }
- }
- $results[$i]['group_name'] = $group_name_list;
- }
- $results[$i]['exe_duration'] = !empty($results[$i]['exe_duration']) ? round($results[$i]['exe_duration'] / 60) : 0;
- $id = $results[$i]['exe_id'];
- $dt = api_convert_and_format_date($results[$i]['exe_weighting']);
- // we filter the results if we have the permission to
- $result_disabled = 0;
- if (isset($results[$i]['results_disabled'])) {
- $result_disabled = (int) $results[$i]['results_disabled'];
- }
- if ($result_disabled == 0) {
- $my_res = $results[$i]['exe_result'];
- $my_total = $results[$i]['exe_weighting'];
- $results[$i]['start_date'] = api_get_local_time($results[$i]['start_date']);
- $results[$i]['exe_date'] = api_get_local_time($results[$i]['exe_date']);
- if (!$results[$i]['propagate_neg'] && $my_res < 0) {
- $my_res = 0;
- }
- $score = self::show_score(
- $my_res,
- $my_total,
- true,
- true,
- false,
- false,
- $decimalSeparator,
- $thousandSeparator,
- $roundValues
- );
- $actions = '<div class="pull-right">';
- if ($is_allowedToEdit) {
- if (isset($teacher_id_list)) {
- if (in_array(
- $results[$i]['exe_user_id'],
- $teacher_id_list
- )) {
- $actions .= Display::return_icon('teacher.png', get_lang('Teacher'));
- }
- }
- $revisedLabel = '';
- switch ($revised) {
- case 0:
- $actions .= "<a href='exercise_show.php?".api_get_cidreq()."&action=qualify&id=$id'>".
- Display:: return_icon(
- 'quiz.png',
- get_lang('Qualify')
- );
- $actions .= '</a>';
- $revisedLabel = Display::label(
- get_lang('NotValidated'),
- 'info'
- );
- break;
- case 1:
- $actions .= "<a href='exercise_show.php?".api_get_cidreq()."&action=edit&id=$id'>".
- Display:: return_icon(
- 'edit.png',
- get_lang('Edit'),
- [],
- ICON_SIZE_SMALL
- );
- $actions .= '</a>';
- $revisedLabel = Display::label(
- get_lang('Validated'),
- 'success'
- );
- break;
- case 2: //finished but not marked as such
- $actions .= '<a href="exercise_report.php?'
- .api_get_cidreq()
- .'&exerciseId='
- .$exercise_id
- .'&a=close&id='
- .$id
- .'">'.
- Display:: return_icon(
- 'lock.png',
- get_lang('MarkAttemptAsClosed'),
- [],
- ICON_SIZE_SMALL
- );
- $actions .= '</a>';
- $revisedLabel = Display::label(
- get_lang('Unclosed'),
- 'warning'
- );
- break;
- case 3: //still ongoing
- $actions .= Display:: return_icon(
- 'clock.png',
- get_lang('AttemptStillOngoingPleaseWait'),
- [],
- ICON_SIZE_SMALL
- );
- $actions .= '';
- $revisedLabel = Display::label(
- get_lang('Ongoing'),
- 'danger'
- );
- break;
- }
- if ($filter == 2) {
- $actions .= ' <a href="exercise_history.php?'.api_get_cidreq().'&exe_id='.$id.'">'.
- Display:: return_icon(
- 'history.png',
- get_lang('ViewHistoryChange')
- ).'</a>';
- }
- // Admin can always delete the attempt
- if (($locked == false || api_is_platform_admin()) && !api_is_student_boss()) {
- $ip = Tracking::get_ip_from_user_event(
- $results[$i]['exe_user_id'],
- api_get_utc_datetime(),
- false
- );
- $actions .= '<a href="http://www.whatsmyip.org/ip-geo-location/?ip='.$ip.'" target="_blank">'
- .Display::return_icon('info.png', $ip)
- .'</a>';
- $recalculateUrl = api_get_path(WEB_CODE_PATH).'exercise/recalculate.php?'.
- api_get_cidreq().'&'.
- http_build_query([
- 'id' => $id,
- 'exercise' => $exercise_id,
- 'user' => $results[$i]['exe_user_id'],
- ]);
- $actions .= Display::url(
- Display::return_icon('reload.png', get_lang('RecalculateResults')),
- $recalculateUrl,
- [
- 'data-exercise' => $exercise_id,
- 'data-user' => $results[$i]['exe_user_id'],
- 'data-id' => $id,
- 'class' => 'exercise-recalculate',
- ]
- );
- $filterByUser = isset($_GET['filter_by_user']) ? (int) $_GET['filter_by_user'] : 0;
- $delete_link = '<a href="exercise_report.php?'.api_get_cidreq().'&filter_by_user='.$filterByUser.'&filter='.$filter.'&exerciseId='.$exercise_id.'&delete=delete&did='.$id.'"
- onclick="javascript:if(!confirm(\''.sprintf(
- addslashes(get_lang('DeleteAttempt')),
- $results[$i]['username'],
- $dt
- ).'\')) return false;">';
- $delete_link .= Display::return_icon(
- 'delete.png',
- addslashes(get_lang('Delete'))
- ).'</a>';
- if (api_is_drh() && !api_is_platform_admin()) {
- $delete_link = null;
- }
- if (api_is_session_admin()) {
- $delete_link = '';
- }
- if ($revised == 3) {
- $delete_link = null;
- }
- $actions .= $delete_link;
- }
- } else {
- $attempt_url = api_get_path(WEB_CODE_PATH).'exercise/result.php?'.api_get_cidreq().'&id='.$results[$i]['exe_id'].'&id_session='.$sessionId;
- $attempt_link = Display::url(
- get_lang('Show'),
- $attempt_url,
- [
- 'class' => 'ajax btn btn-default',
- 'data-title' => get_lang('Show'),
- ]
- );
- $actions .= $attempt_link;
- }
- $actions .= '</div>';
- if (!empty($userExtraFieldsToAdd)) {
- foreach ($userExtraFieldsToAdd as $variable) {
- $extraFieldValue = new ExtraFieldValue('user');
- $values = $extraFieldValue->get_values_by_handler_and_field_variable(
- $results[$i]['user_id'],
- $variable
- );
- if (isset($values['value'])) {
- $results[$i][$variable] = $values['value'];
- }
- }
- }
- $exeId = $results[$i]['exe_id'];
- $results[$i]['id'] = $exeId;
- $category_list = [];
- if ($is_allowedToEdit) {
- $sessionName = '';
- $sessionStartAccessDate = '';
- if (!empty($results[$i]['session_id'])) {
- $sessionInfo = api_get_session_info($results[$i]['session_id']);
- if (!empty($sessionInfo)) {
- $sessionName = $sessionInfo['name'];
- $sessionStartAccessDate = api_get_local_time($sessionInfo['access_start_date']);
- }
- }
- $objExercise = new Exercise($course_id);
- if ($showExerciseCategories) {
- // Getting attempt info
- $exercise_stat_info = $objExercise->get_stat_track_exercise_info_by_exe_id($exeId);
- if (!empty($exercise_stat_info['data_tracking'])) {
- $question_list = explode(',', $exercise_stat_info['data_tracking']);
- if (!empty($question_list)) {
- foreach ($question_list as $questionId) {
- $objQuestionTmp = Question::read($questionId, $objExercise->course);
- // We're inside *one* question. Go through each possible answer for this question
- $result = $objExercise->manage_answer(
- $exeId,
- $questionId,
- null,
- 'exercise_result',
- false,
- false,
- true,
- false,
- $objExercise->selectPropagateNeg(),
- null,
- true
- );
- $my_total_score = $result['score'];
- $my_total_weight = $result['weight'];
- // Category report
- $category_was_added_for_this_test = false;
- if (isset($objQuestionTmp->category) && !empty($objQuestionTmp->category)) {
- if (!isset($category_list[$objQuestionTmp->category]['score'])) {
- $category_list[$objQuestionTmp->category]['score'] = 0;
- }
- if (!isset($category_list[$objQuestionTmp->category]['total'])) {
- $category_list[$objQuestionTmp->category]['total'] = 0;
- }
- $category_list[$objQuestionTmp->category]['score'] += $my_total_score;
- $category_list[$objQuestionTmp->category]['total'] += $my_total_weight;
- $category_was_added_for_this_test = true;
- }
- if (isset($objQuestionTmp->category_list) &&
- !empty($objQuestionTmp->category_list)
- ) {
- foreach ($objQuestionTmp->category_list as $category_id) {
- $category_list[$category_id]['score'] += $my_total_score;
- $category_list[$category_id]['total'] += $my_total_weight;
- $category_was_added_for_this_test = true;
- }
- }
- // No category for this question!
- if ($category_was_added_for_this_test == false) {
- if (!isset($category_list['none']['score'])) {
- $category_list['none']['score'] = 0;
- }
- if (!isset($category_list['none']['total'])) {
- $category_list['none']['total'] = 0;
- }
- $category_list['none']['score'] += $my_total_score;
- $category_list['none']['total'] += $my_total_weight;
- }
- }
- }
- }
- }
- foreach ($category_list as $categoryId => $result) {
- $scoreToDisplay = self::show_score(
- $result['score'],
- $result['total'],
- true,
- true,
- false,
- false,
- $decimalSeparator,
- $thousandSeparator,
- $roundValues
- );
- $results[$i]['category_'.$categoryId] = $scoreToDisplay;
- $results[$i]['category_'.$categoryId.'_score_percentage'] = self::show_score(
- $result['score'],
- $result['total'],
- true,
- true,
- true, // $show_only_percentage = false
- true, // hide % sign
- $decimalSeparator,
- $thousandSeparator,
- $roundValues
- );
- $results[$i]['category_'.$categoryId.'_only_score'] = $result['score'];
- $results[$i]['category_'.$categoryId.'_total'] = $result['total'];
- }
- $results[$i]['session'] = $sessionName;
- $results[$i]['session_access_start_date'] = $sessionStartAccessDate;
- $results[$i]['status'] = $revisedLabel;
- $results[$i]['score'] = $score;
- $results[$i]['score_percentage'] = self::show_score(
- $my_res,
- $my_total,
- true,
- true,
- true,
- true,
- $decimalSeparator,
- $thousandSeparator,
- $roundValues
- );
- if ($roundValues) {
- $whole = floor($my_res); // 1
- $fraction = $my_res - $whole; // .25
- if ($fraction >= 0.5) {
- $onlyScore = ceil($my_res);
- } else {
- $onlyScore = round($my_res);
- }
- } else {
- $onlyScore = $scoreDisplay->format_score(
- $my_res,
- false,
- $decimalSeparator,
- $thousandSeparator
- );
- }
- $results[$i]['only_score'] = $onlyScore;
- if ($roundValues) {
- $whole = floor($my_total); // 1
- $fraction = $my_total - $whole; // .25
- if ($fraction >= 0.5) {
- $onlyTotal = ceil($my_total);
- } else {
- $onlyTotal = round($my_total);
- }
- } else {
- $onlyTotal = $scoreDisplay->format_score(
- $my_total,
- false,
- $decimalSeparator,
- $thousandSeparator
- );
- }
- $results[$i]['total'] = $onlyTotal;
- $results[$i]['lp'] = $lp_name;
- $results[$i]['actions'] = $actions;
- $listInfo[] = $results[$i];
- } else {
- $results[$i]['status'] = $revisedLabel;
- $results[$i]['score'] = $score;
- $results[$i]['actions'] = $actions;
- $listInfo[] = $results[$i];
- }
- }
- }
- }
- } else {
- $hpresults = [];
- $res = Database::query($hpsql);
- if ($res !== false) {
- $i = 0;
- while ($resA = Database::fetch_array($res, 'NUM')) {
- for ($j = 0; $j < 6; $j++) {
- $hpresults[$i][$j] = $resA[$j];
- }
- $i++;
- }
- }
- // Print HotPotatoes test results.
- if (is_array($hpresults)) {
- for ($i = 0; $i < count($hpresults); $i++) {
- $hp_title = GetQuizName($hpresults[$i][3], $documentPath);
- if ($hp_title == '') {
- $hp_title = basename($hpresults[$i][3]);
- }
- $hp_date = api_get_local_time(
- $hpresults[$i][6],
- null,
- date_default_timezone_get()
- );
- $hp_result = round(($hpresults[$i][4] / ($hpresults[$i][5] != 0 ? $hpresults[$i][5] : 1)) * 100, 2);
- $hp_result .= '% ('.$hpresults[$i][4].' / '.$hpresults[$i][5].')';
- if ($is_allowedToEdit) {
- $listInfo[] = [
- $hpresults[$i][0],
- $hpresults[$i][1],
- $hpresults[$i][2],
- '',
- $hp_title,
- '-',
- $hp_date,
- $hp_result,
- '-',
- ];
- } else {
- $listInfo[] = [
- $hp_title,
- '-',
- $hp_date,
- $hp_result,
- '-',
- ];
- }
- }
- }
- }
- return $listInfo;
- }
- /**
- * @param $score
- * @param $weight
- *
- * @return array
- */
- public static function convertScoreToPlatformSetting($score, $weight)
- {
- $maxNote = api_get_setting('exercise_max_score');
- $minNote = api_get_setting('exercise_min_score');
- if ($maxNote != '' && $minNote != '') {
- if (!empty($weight) && intval($weight) != 0) {
- $score = $minNote + ($maxNote - $minNote) * $score / $weight;
- } else {
- $score = $minNote;
- }
- $weight = $maxNote;
- }
- return ['score' => $score, 'weight' => $weight];
- }
- /**
- * Converts the score with the exercise_max_note and exercise_min_score
- * the platform settings + formats the results using the float_format function.
- *
- * @param float $score
- * @param float $weight
- * @param bool $show_percentage show percentage or not
- * @param bool $use_platform_settings use or not the platform settings
- * @param bool $show_only_percentage
- * @param bool $hidePercentageSign hide "%" sign
- * @param string $decimalSeparator
- * @param string $thousandSeparator
- * @param bool $roundValues This option rounds the float values into a int using ceil()
- *
- * @return string an html with the score modified
- */
- public static function show_score(
- $score,
- $weight,
- $show_percentage = true,
- $use_platform_settings = true,
- $show_only_percentage = false,
- $hidePercentageSign = false,
- $decimalSeparator = '.',
- $thousandSeparator = ',',
- $roundValues = false
- ) {
- if (is_null($score) && is_null($weight)) {
- return '-';
- }
- if ($use_platform_settings) {
- $result = self::convertScoreToPlatformSetting($score, $weight);
- $score = $result['score'];
- $weight = $result['weight'];
- }
- $percentage = (100 * $score) / ($weight != 0 ? $weight : 1);
- // Formats values
- $percentage = float_format($percentage, 1);
- $score = float_format($score, 1);
- $weight = float_format($weight, 1);
- if ($roundValues) {
- $whole = floor($percentage); // 1
- $fraction = $percentage - $whole; // .25
- // Formats values
- if ($fraction >= 0.5) {
- $percentage = ceil($percentage);
- } else {
- $percentage = round($percentage);
- }
- $whole = floor($score); // 1
- $fraction = $score - $whole; // .25
- if ($fraction >= 0.5) {
- $score = ceil($score);
- } else {
- $score = round($score);
- }
- $whole = floor($weight); // 1
- $fraction = $weight - $whole; // .25
- if ($fraction >= 0.5) {
- $weight = ceil($weight);
- } else {
- $weight = round($weight);
- }
- } else {
- // Formats values
- $percentage = float_format($percentage, 1, $decimalSeparator, $thousandSeparator);
- $score = float_format($score, 1, $decimalSeparator, $thousandSeparator);
- $weight = float_format($weight, 1, $decimalSeparator, $thousandSeparator);
- }
- if ($show_percentage) {
- $percentageSign = '%';
- if ($hidePercentageSign) {
- $percentageSign = '';
- }
- $html = $percentage."$percentageSign ($score / $weight)";
- if ($show_only_percentage) {
- $html = $percentage.$percentageSign;
- }
- } else {
- $html = $score.' / '.$weight;
- }
- // Over write score
- $scoreBasedInModel = self::convertScoreToModel($percentage);
- if (!empty($scoreBasedInModel)) {
- $html = $scoreBasedInModel;
- }
- // Ignore other formats and use the configuratio['exercise_score_format'] value
- // But also keep the round values settings.
- $format = api_get_configuration_value('exercise_score_format');
- if (!empty($format)) {
- $html = ScoreDisplay::instance()->display_score([$score, $weight], $format);
- }
- $html = Display::span($html, ['class' => 'score_exercise']);
- return $html;
- }
- /**
- * @param array $model
- * @param float $percentage
- *
- * @return string
- */
- public static function getModelStyle($model, $percentage)
- {
- $modelWithStyle = '<span class="'.$model['css_class'].'"> </span>';
- return $modelWithStyle;
- }
- /**
- * @param float $percentage value between 0 and 100
- *
- * @return string
- */
- public static function convertScoreToModel($percentage)
- {
- $model = self::getCourseScoreModel();
- if (!empty($model)) {
- $scoreWithGrade = [];
- foreach ($model['score_list'] as $item) {
- if ($percentage >= $item['min'] && $percentage <= $item['max']) {
- $scoreWithGrade = $item;
- break;
- }
- }
- if (!empty($scoreWithGrade)) {
- return self::getModelStyle($scoreWithGrade, $percentage);
- }
- }
- return '';
- }
- /**
- * @return array
- */
- public static function getCourseScoreModel()
- {
- $modelList = self::getScoreModels();
- if (empty($modelList)) {
- return [];
- }
- $courseInfo = api_get_course_info();
- if (!empty($courseInfo)) {
- $scoreModelId = api_get_course_setting('score_model_id');
- if ($scoreModelId != -1) {
- $modelIdList = array_column($modelList['models'], 'id');
- if (in_array($scoreModelId, $modelIdList)) {
- foreach ($modelList['models'] as $item) {
- if ($item['id'] == $scoreModelId) {
- return $item;
- }
- }
- }
- }
- }
- return [];
- }
- /**
- * @return array
- */
- public static function getScoreModels()
- {
- return api_get_configuration_value('score_grade_model');
- }
- /**
- * @param float $score
- * @param float $weight
- * @param string $pass_percentage
- *
- * @return bool
- */
- public static function isSuccessExerciseResult($score, $weight, $pass_percentage)
- {
- $percentage = float_format(
- ($score / ($weight != 0 ? $weight : 1)) * 100,
- 1
- );
- if (isset($pass_percentage) && !empty($pass_percentage)) {
- if ($percentage >= $pass_percentage) {
- return true;
- }
- }
- return false;
- }
- /**
- * @param FormValidator $form
- * @param string $name
- * @param $weight
- * @param $selected
- *
- * @return bool
- */
- public static function addScoreModelInput(
- FormValidator $form,
- $name,
- $weight,
- $selected
- ) {
- $model = self::getCourseScoreModel();
- if (empty($model)) {
- return false;
- }
- /** @var HTML_QuickForm_select $element */
- $element = $form->createElement(
- 'select',
- $name,
- get_lang('Qualification'),
- [],
- ['class' => 'exercise_mark_select']
- );
- foreach ($model['score_list'] as $item) {
- $i = api_number_format($item['score_to_qualify'] / 100 * $weight, 2);
- $label = self::getModelStyle($item, $i);
- $attributes = [
- 'class' => $item['css_class'],
- ];
- if ($selected == $i) {
- $attributes['selected'] = 'selected';
- }
- $element->addOption($label, $i, $attributes);
- }
- $form->addElement($element);
- }
- /**
- * @return string
- */
- public static function getJsCode()
- {
- // Filling the scores with the right colors.
- $models = self::getCourseScoreModel();
- $cssListToString = '';
- if (!empty($models)) {
- $cssList = array_column($models['score_list'], 'css_class');
- $cssListToString = implode(' ', $cssList);
- }
- if (empty($cssListToString)) {
- return '';
- }
- $js = <<<EOT
-
- function updateSelect(element) {
- var spanTag = element.parent().find('span.filter-option');
- var value = element.val();
- var selectId = element.attr('id');
- var optionClass = $('#' + selectId + ' option[value="'+value+'"]').attr('class');
- spanTag.removeClass('$cssListToString');
- spanTag.addClass(optionClass);
- }
-
- $(function() {
- // Loading values
- $('.exercise_mark_select').on('loaded.bs.select', function() {
- updateSelect($(this));
- });
- // On change
- $('.exercise_mark_select').on('changed.bs.select', function() {
- updateSelect($(this));
- });
- });
- EOT;
- return $js;
- }
- /**
- * @param float $score
- * @param float $weight
- * @param string $pass_percentage
- *
- * @return string
- */
- public static function showSuccessMessage($score, $weight, $pass_percentage)
- {
- $res = '';
- if (self::isPassPercentageEnabled($pass_percentage)) {
- $isSuccess = self::isSuccessExerciseResult(
- $score,
- $weight,
- $pass_percentage
- );
- if ($isSuccess) {
- $html = get_lang('CongratulationsYouPassedTheTest');
- $icon = Display::return_icon(
- 'completed.png',
- get_lang('Correct'),
- [],
- ICON_SIZE_MEDIUM
- );
- } else {
- $html = get_lang('YouDidNotReachTheMinimumScore');
- $icon = Display::return_icon(
- 'warning.png',
- get_lang('Wrong'),
- [],
- ICON_SIZE_MEDIUM
- );
- }
- $html = Display::tag('h4', $html);
- $html .= Display::tag(
- 'h5',
- $icon,
- ['style' => 'width:40px; padding:2px 10px 0px 0px']
- );
- $res = $html;
- }
- return $res;
- }
- /**
- * Return true if pass_pourcentage activated (we use the pass pourcentage feature
- * return false if pass_percentage = 0 (we don't use the pass pourcentage feature.
- *
- * @param $value
- *
- * @return bool
- * In this version, pass_percentage and show_success_message are disabled if
- * pass_percentage is set to 0
- */
- public static function isPassPercentageEnabled($value)
- {
- return $value > 0;
- }
- /**
- * Converts a numeric value in a percentage example 0.66666 to 66.67 %.
- *
- * @param $value
- *
- * @return float Converted number
- */
- public static function convert_to_percentage($value)
- {
- $return = '-';
- if ($value != '') {
- $return = float_format($value * 100, 1).' %';
- }
- return $return;
- }
- /**
- * Getting all active exercises from a course from a session
- * (if a session_id is provided we will show all the exercises in the course +
- * all exercises in the session).
- *
- * @param array $course_info
- * @param int $session_id
- * @param bool $check_publication_dates
- * @param string $search Search exercise name
- * @param bool $search_all_sessions Search exercises in all sessions
- * @param int 0 = only inactive exercises
- * 1 = only active exercises,
- * 2 = all exercises
- * 3 = active <> -1
- *
- * @return array array with exercise data
- */
- public static function get_all_exercises(
- $course_info = null,
- $session_id = 0,
- $check_publication_dates = false,
- $search = '',
- $search_all_sessions = false,
- $active = 2
- ) {
- $course_id = api_get_course_int_id();
- if (!empty($course_info) && !empty($course_info['real_id'])) {
- $course_id = $course_info['real_id'];
- }
- if ($session_id == -1) {
- $session_id = 0;
- }
- $now = api_get_utc_datetime();
- $timeConditions = '';
- if ($check_publication_dates) {
- // Start and end are set
- $timeConditions = " AND ((start_time <> '' AND start_time < '$now' AND end_time <> '' AND end_time > '$now' ) OR ";
- // only start is set
- $timeConditions .= " (start_time <> '' AND start_time < '$now' AND end_time is NULL) OR ";
- // only end is set
- $timeConditions .= " (start_time IS NULL AND end_time <> '' AND end_time > '$now') OR ";
- // nothing is set
- $timeConditions .= ' (start_time IS NULL AND end_time IS NULL)) ';
- }
- $needle_where = !empty($search) ? " AND title LIKE '?' " : '';
- $needle = !empty($search) ? "%".$search."%" : '';
- // Show courses by active status
- $active_sql = '';
- if ($active == 3) {
- $active_sql = ' active <> -1 AND';
- } else {
- if ($active != 2) {
- $active_sql = sprintf(' active = %d AND', $active);
- }
- }
- if ($search_all_sessions == true) {
- $conditions = [
- 'where' => [
- $active_sql.' c_id = ? '.$needle_where.$timeConditions => [
- $course_id,
- $needle,
- ],
- ],
- 'order' => 'title',
- ];
- } else {
- if (empty($session_id)) {
- $conditions = [
- 'where' => [
- $active_sql.' (session_id = 0 OR session_id IS NULL) AND c_id = ? '.$needle_where.$timeConditions => [
- $course_id,
- $needle,
- ],
- ],
- 'order' => 'title',
- ];
- } else {
- $conditions = [
- 'where' => [
- $active_sql.' (session_id = 0 OR session_id IS NULL OR session_id = ? ) AND c_id = ? '.$needle_where.$timeConditions => [
- $session_id,
- $course_id,
- $needle,
- ],
- ],
- 'order' => 'title',
- ];
- }
- }
- $table = Database::get_course_table(TABLE_QUIZ_TEST);
- return Database::select('*', $table, $conditions);
- }
- /**
- * Getting all exercises (active only or all)
- * from a course from a session
- * (if a session_id is provided we will show all the exercises in the
- * course + all exercises in the session).
- *
- * @param array course data
- * @param int session id
- * @param int course c_id
- * @param bool $only_active_exercises
- *
- * @return array array with exercise data
- * modified by Hubert Borderiou
- */
- public static function get_all_exercises_for_course_id(
- $course_info = null,
- $session_id = 0,
- $course_id = 0,
- $only_active_exercises = true
- ) {
- $table = Database::get_course_table(TABLE_QUIZ_TEST);
- if ($only_active_exercises) {
- // Only active exercises.
- $sql_active_exercises = "active = 1 AND ";
- } else {
- // Not only active means visible and invisible NOT deleted (-2)
- $sql_active_exercises = "active IN (1, 0) AND ";
- }
- if ($session_id == -1) {
- $session_id = 0;
- }
- $params = [
- $session_id,
- $course_id,
- ];
- if (empty($session_id)) {
- $conditions = [
- 'where' => ["$sql_active_exercises (session_id = 0 OR session_id IS NULL) AND c_id = ?" => [$course_id]],
- 'order' => 'title',
- ];
- } else {
- // All exercises
- $conditions = [
- 'where' => ["$sql_active_exercises (session_id = 0 OR session_id IS NULL OR session_id = ? ) AND c_id=?" => $params],
- 'order' => 'title',
- ];
- }
- return Database::select('*', $table, $conditions);
- }
- /**
- * Gets the position of the score based in a given score (result/weight)
- * and the exe_id based in the user list
- * (NO Exercises in LPs ).
- *
- * @param float $my_score user score to be compared *attention*
- * $my_score = score/weight and not just the score
- * @param int $my_exe_id exe id of the exercise
- * (this is necessary because if 2 students have the same score the one
- * with the minor exe_id will have a best position, just to be fair and FIFO)
- * @param int $exercise_id
- * @param string $course_code
- * @param int $session_id
- * @param array $user_list
- * @param bool $return_string
- *
- * @return int the position of the user between his friends in a course
- * (or course within a session)
- */
- public static function get_exercise_result_ranking(
- $my_score,
- $my_exe_id,
- $exercise_id,
- $course_code,
- $session_id = 0,
- $user_list = [],
- $return_string = true
- ) {
- //No score given we return
- if (is_null($my_score)) {
- return '-';
- }
- if (empty($user_list)) {
- return '-';
- }
- $best_attempts = [];
- foreach ($user_list as $user_data) {
- $user_id = $user_data['user_id'];
- $best_attempts[$user_id] = self::get_best_attempt_by_user(
- $user_id,
- $exercise_id,
- $course_code,
- $session_id
- );
- }
- if (empty($best_attempts)) {
- return 1;
- } else {
- $position = 1;
- $my_ranking = [];
- foreach ($best_attempts as $user_id => $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $my_ranking[$user_id] = $result['exe_result'] / $result['exe_weighting'];
- } else {
- $my_ranking[$user_id] = 0;
- }
- }
- //if (!empty($my_ranking)) {
- asort($my_ranking);
- $position = count($my_ranking);
- if (!empty($my_ranking)) {
- foreach ($my_ranking as $user_id => $ranking) {
- if ($my_score >= $ranking) {
- if ($my_score == $ranking && isset($best_attempts[$user_id]['exe_id'])) {
- $exe_id = $best_attempts[$user_id]['exe_id'];
- if ($my_exe_id < $exe_id) {
- $position--;
- }
- } else {
- $position--;
- }
- }
- }
- }
- //}
- $return_value = [
- 'position' => $position,
- 'count' => count($my_ranking),
- ];
- if ($return_string) {
- if (!empty($position) && !empty($my_ranking)) {
- $return_value = $position.'/'.count($my_ranking);
- } else {
- $return_value = '-';
- }
- }
- return $return_value;
- }
- }
- /**
- * Gets the position of the score based in a given score (result/weight) and the exe_id based in all attempts
- * (NO Exercises in LPs ) old functionality by attempt.
- *
- * @param float user score to be compared attention => score/weight
- * @param int exe id of the exercise
- * (this is necessary because if 2 students have the same score the one
- * with the minor exe_id will have a best position, just to be fair and FIFO)
- * @param int exercise id
- * @param string course code
- * @param int session id
- * @param bool $return_string
- *
- * @return int the position of the user between his friends in a course (or course within a session)
- */
- public static function get_exercise_result_ranking_by_attempt(
- $my_score,
- $my_exe_id,
- $exercise_id,
- $courseId,
- $session_id = 0,
- $return_string = true
- ) {
- if (empty($session_id)) {
- $session_id = 0;
- }
- if (is_null($my_score)) {
- return '-';
- }
- $user_results = Event::get_all_exercise_results(
- $exercise_id,
- $courseId,
- $session_id,
- false
- );
- $position_data = [];
- if (empty($user_results)) {
- return 1;
- } else {
- $position = 1;
- $my_ranking = [];
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $my_ranking[$result['exe_id']] = $result['exe_result'] / $result['exe_weighting'];
- } else {
- $my_ranking[$result['exe_id']] = 0;
- }
- }
- asort($my_ranking);
- $position = count($my_ranking);
- if (!empty($my_ranking)) {
- foreach ($my_ranking as $exe_id => $ranking) {
- if ($my_score >= $ranking) {
- if ($my_score == $ranking) {
- if ($my_exe_id < $exe_id) {
- $position--;
- }
- } else {
- $position--;
- }
- }
- }
- }
- $return_value = [
- 'position' => $position,
- 'count' => count($my_ranking),
- ];
- if ($return_string) {
- if (!empty($position) && !empty($my_ranking)) {
- return $position.'/'.count($my_ranking);
- }
- }
- return $return_value;
- }
- }
- /**
- * Get the best attempt in a exercise (NO Exercises in LPs ).
- *
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- *
- * @return array
- */
- public static function get_best_attempt_in_course($exercise_id, $courseId, $session_id)
- {
- $user_results = Event::get_all_exercise_results(
- $exercise_id,
- $courseId,
- $session_id,
- false
- );
- $best_score_data = [];
- $best_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) &&
- intval($result['exe_weighting']) != 0
- ) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- if ($score >= $best_score) {
- $best_score = $score;
- $best_score_data = $result;
- }
- }
- }
- }
- return $best_score_data;
- }
- /**
- * Get the best score in a exercise (NO Exercises in LPs ).
- *
- * @param int $user_id
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- *
- * @return array
- */
- public static function get_best_attempt_by_user(
- $user_id,
- $exercise_id,
- $courseId,
- $session_id
- ) {
- $user_results = Event::get_all_exercise_results(
- $exercise_id,
- $courseId,
- $session_id,
- false,
- $user_id
- );
- $best_score_data = [];
- $best_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- if ($score >= $best_score) {
- $best_score = $score;
- $best_score_data = $result;
- }
- }
- }
- }
- return $best_score_data;
- }
- /**
- * Get average score (NO Exercises in LPs ).
- *
- * @param int exercise id
- * @param int $courseId
- * @param int session id
- *
- * @return float Average score
- */
- public static function get_average_score($exercise_id, $courseId, $session_id)
- {
- $user_results = Event::get_all_exercise_results(
- $exercise_id,
- $courseId,
- $session_id
- );
- $avg_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- $avg_score += $score;
- }
- }
- $avg_score = float_format($avg_score / count($user_results), 1);
- }
- return $avg_score;
- }
- /**
- * Get average score by score (NO Exercises in LPs ).
- *
- * @param int $courseId
- * @param int session id
- *
- * @return float Average score
- */
- public static function get_average_score_by_course($courseId, $session_id)
- {
- $user_results = Event::get_all_exercise_results_by_course(
- $courseId,
- $session_id,
- false
- );
- $avg_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval(
- $result['exe_weighting']
- ) != 0
- ) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- $avg_score += $score;
- }
- }
- // We assume that all exe_weighting
- $avg_score = $avg_score / count($user_results);
- }
- return $avg_score;
- }
- /**
- * @param int $user_id
- * @param int $courseId
- * @param int $session_id
- *
- * @return float|int
- */
- public static function get_average_score_by_course_by_user(
- $user_id,
- $courseId,
- $session_id
- ) {
- $user_results = Event::get_all_exercise_results_by_user(
- $user_id,
- $courseId,
- $session_id
- );
- $avg_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- $avg_score += $score;
- }
- }
- // We assume that all exe_weighting
- $avg_score = ($avg_score / count($user_results));
- }
- return $avg_score;
- }
- /**
- * Get average score by score (NO Exercises in LPs ).
- *
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- * @param int $user_count
- *
- * @return float Best average score
- */
- public static function get_best_average_score_by_exercise(
- $exercise_id,
- $courseId,
- $session_id,
- $user_count
- ) {
- $user_results = Event::get_best_exercise_results_by_user(
- $exercise_id,
- $courseId,
- $session_id
- );
- $avg_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- $avg_score += $score;
- }
- }
- // We asumme that all exe_weighting
- if (!empty($user_count)) {
- $avg_score = float_format($avg_score / $user_count, 1) * 100;
- } else {
- $avg_score = 0;
- }
- }
- return $avg_score;
- }
- /**
- * Get average score by score (NO Exercises in LPs ).
- *
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- *
- * @return float Best average score
- */
- public static function getBestScoreByExercise(
- $exercise_id,
- $courseId,
- $session_id
- ) {
- $user_results = Event::get_best_exercise_results_by_user(
- $exercise_id,
- $courseId,
- $session_id
- );
- $avg_score = 0;
- if (!empty($user_results)) {
- foreach ($user_results as $result) {
- if (!empty($result['exe_weighting']) && intval($result['exe_weighting']) != 0) {
- $score = $result['exe_result'] / $result['exe_weighting'];
- $avg_score += $score;
- }
- }
- }
- return $avg_score;
- }
- /**
- * @param string $course_code
- * @param int $session_id
- *
- * @return array
- */
- public static function get_exercises_to_be_taken($course_code, $session_id)
- {
- $course_info = api_get_course_info($course_code);
- $exercises = self::get_all_exercises($course_info, $session_id);
- $result = [];
- $now = time() + 15 * 24 * 60 * 60;
- foreach ($exercises as $exercise_item) {
- if (isset($exercise_item['end_time']) &&
- !empty($exercise_item['end_time']) &&
- api_strtotime($exercise_item['end_time'], 'UTC') < $now
- ) {
- $result[] = $exercise_item;
- }
- }
- return $result;
- }
- /**
- * Get student results (only in completed exercises) stats by question.
- *
- * @param int $question_id
- * @param int $exercise_id
- * @param string $course_code
- * @param int $session_id
- *
- * @return array
- */
- public static function get_student_stats_by_question(
- $question_id,
- $exercise_id,
- $course_code,
- $session_id
- ) {
- $track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $question_id = (int) $question_id;
- $exercise_id = (int) $exercise_id;
- $course_code = Database::escape_string($course_code);
- $session_id = (int) $session_id;
- $courseId = api_get_course_int_id($course_code);
- $sql = "SELECT MAX(marks) as max, MIN(marks) as min, AVG(marks) as average
- FROM $track_exercises e
- INNER JOIN $track_attempt a
- ON (
- a.exe_id = e.exe_id AND
- e.c_id = a.c_id AND
- e.session_id = a.session_id
- )
- WHERE
- exe_exo_id = $exercise_id AND
- a.c_id = $courseId AND
- e.session_id = $session_id AND
- question_id = $question_id AND
- status = ''
- LIMIT 1";
- $result = Database::query($sql);
- $return = [];
- if ($result) {
- $return = Database::fetch_array($result, 'ASSOC');
- }
- return $return;
- }
- /**
- * Get the correct answer count for a fill blanks question.
- *
- * @param int $question_id
- * @param int $exercise_id
- *
- * @return array
- */
- public static function getNumberStudentsFillBlanksAnswerCount(
- $question_id,
- $exercise_id
- ) {
- $listStudentsId = [];
- $listAllStudentInfo = CourseManager::get_student_list_from_course_code(
- api_get_course_id(),
- true
- );
- foreach ($listAllStudentInfo as $i => $listStudentInfo) {
- $listStudentsId[] = $listStudentInfo['user_id'];
- }
- $listFillTheBlankResult = FillBlanks::getFillTheBlankResult(
- $exercise_id,
- $question_id,
- $listStudentsId,
- '1970-01-01',
- '3000-01-01'
- );
- $arrayCount = [];
- foreach ($listFillTheBlankResult as $resultCount) {
- foreach ($resultCount as $index => $count) {
- //this is only for declare the array index per answer
- $arrayCount[$index] = 0;
- }
- }
- foreach ($listFillTheBlankResult as $resultCount) {
- foreach ($resultCount as $index => $count) {
- $count = ($count === 0) ? 1 : 0;
- $arrayCount[$index] += $count;
- }
- }
- return $arrayCount;
- }
- /**
- * Get the number of questions with answers.
- *
- * @param int $question_id
- * @param int $exercise_id
- * @param string $course_code
- * @param int $session_id
- * @param string $questionType
- *
- * @return int
- */
- public static function get_number_students_question_with_answer_count(
- $question_id,
- $exercise_id,
- $course_code,
- $session_id,
- $questionType = ''
- ) {
- $track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseUser = Database::get_main_table(TABLE_MAIN_COURSE_USER);
- $courseTable = Database::get_main_table(TABLE_MAIN_COURSE);
- $courseUserSession = Database::get_main_table(TABLE_MAIN_SESSION_COURSE_USER);
- $question_id = intval($question_id);
- $exercise_id = intval($exercise_id);
- $courseId = api_get_course_int_id($course_code);
- $session_id = intval($session_id);
- if ($questionType == FILL_IN_BLANKS) {
- $listStudentsId = [];
- $listAllStudentInfo = CourseManager::get_student_list_from_course_code(
- api_get_course_id(),
- true
- );
- foreach ($listAllStudentInfo as $i => $listStudentInfo) {
- $listStudentsId[] = $listStudentInfo['user_id'];
- }
- $listFillTheBlankResult = FillBlanks::getFillTheBlankResult(
- $exercise_id,
- $question_id,
- $listStudentsId,
- '1970-01-01',
- '3000-01-01'
- );
- return FillBlanks::getNbResultFillBlankAll($listFillTheBlankResult);
- }
- if (empty($session_id)) {
- $courseCondition = "
- INNER JOIN $courseUser cu
- ON cu.c_id = c.id AND cu.user_id = exe_user_id";
- $courseConditionWhere = " AND relation_type <> 2 AND cu.status = ".STUDENT;
- } else {
- $courseCondition = "
- INNER JOIN $courseUserSession cu
- ON cu.c_id = c.id AND cu.user_id = exe_user_id";
- $courseConditionWhere = " AND cu.status = 0 ";
- }
- $sql = "SELECT DISTINCT exe_user_id
- FROM $track_exercises e
- INNER JOIN $track_attempt a
- ON (
- a.exe_id = e.exe_id AND
- e.c_id = a.c_id AND
- e.session_id = a.session_id
- )
- INNER JOIN $courseTable c
- ON (c.id = a.c_id)
- $courseCondition
- WHERE
- exe_exo_id = $exercise_id AND
- a.c_id = $courseId AND
- e.session_id = $session_id AND
- question_id = $question_id AND
- answer <> '0' AND
- e.status = ''
- $courseConditionWhere
- ";
- $result = Database::query($sql);
- $return = 0;
- if ($result) {
- $return = Database::num_rows($result);
- }
- return $return;
- }
- /**
- * Get number of answers to hotspot questions.
- *
- * @param int $answer_id
- * @param int $question_id
- * @param int $exercise_id
- * @param string $course_code
- * @param int $session_id
- *
- * @return int
- */
- public static function get_number_students_answer_hotspot_count(
- $answer_id,
- $question_id,
- $exercise_id,
- $course_code,
- $session_id
- ) {
- $track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $track_hotspot = Database::get_main_table(TABLE_STATISTIC_TRACK_E_HOTSPOT);
- $courseUser = Database::get_main_table(TABLE_MAIN_COURSE_USER);
- $courseTable = Database::get_main_table(TABLE_MAIN_COURSE);
- $courseUserSession = Database::get_main_table(TABLE_MAIN_SESSION_COURSE_USER);
- $question_id = (int) $question_id;
- $answer_id = (int) $answer_id;
- $exercise_id = (int) $exercise_id;
- $course_code = Database::escape_string($course_code);
- $session_id = (int) $session_id;
- if (empty($session_id)) {
- $courseCondition = "
- INNER JOIN $courseUser cu
- ON cu.c_id = c.id AND cu.user_id = exe_user_id";
- $courseConditionWhere = " AND relation_type <> 2 AND cu.status = ".STUDENT;
- } else {
- $courseCondition = "
- INNER JOIN $courseUserSession cu
- ON cu.c_id = c.id AND cu.user_id = exe_user_id";
- $courseConditionWhere = ' AND cu.status = 0 ';
- }
- $sql = "SELECT DISTINCT exe_user_id
- FROM $track_exercises e
- INNER JOIN $track_hotspot a
- ON (a.hotspot_exe_id = e.exe_id)
- INNER JOIN $courseTable c
- ON (hotspot_course_code = c.code)
- $courseCondition
- WHERE
- exe_exo_id = $exercise_id AND
- a.hotspot_course_code = '$course_code' AND
- e.session_id = $session_id AND
- hotspot_answer_id = $answer_id AND
- hotspot_question_id = $question_id AND
- hotspot_correct = 1 AND
- e.status = ''
- $courseConditionWhere
- ";
- $result = Database::query($sql);
- $return = 0;
- if ($result) {
- $return = Database::num_rows($result);
- }
- return $return;
- }
- /**
- * @param int $answer_id
- * @param int $question_id
- * @param int $exercise_id
- * @param string $course_code
- * @param int $session_id
- * @param string $question_type
- * @param string $correct_answer
- * @param string $current_answer
- *
- * @return int
- */
- public static function get_number_students_answer_count(
- $answer_id,
- $question_id,
- $exercise_id,
- $course_code,
- $session_id,
- $question_type = null,
- $correct_answer = null,
- $current_answer = null
- ) {
- $track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseTable = Database::get_main_table(TABLE_MAIN_COURSE);
- $courseUser = Database::get_main_table(TABLE_MAIN_COURSE_USER);
- $courseUserSession = Database::get_main_table(TABLE_MAIN_SESSION_COURSE_USER);
- $question_id = (int) $question_id;
- $answer_id = (int) $answer_id;
- $exercise_id = (int) $exercise_id;
- $courseId = api_get_course_int_id($course_code);
- $session_id = (int) $session_id;
- switch ($question_type) {
- case FILL_IN_BLANKS:
- $answer_condition = '';
- $select_condition = ' e.exe_id, answer ';
- break;
- case MATCHING:
- case MATCHING_DRAGGABLE:
- default:
- $answer_condition = " answer = $answer_id AND ";
- $select_condition = ' DISTINCT exe_user_id ';
- }
- if (empty($session_id)) {
- $courseCondition = "
- INNER JOIN $courseUser cu
- ON cu.c_id = c.id AND cu.user_id = exe_user_id";
- $courseConditionWhere = " AND relation_type <> 2 AND cu.status = ".STUDENT;
- } else {
- $courseCondition = "
- INNER JOIN $courseUserSession cu
- ON cu.c_id = a.c_id AND cu.user_id = exe_user_id";
- $courseConditionWhere = ' AND cu.status = 0 ';
- }
- $sql = "SELECT $select_condition
- FROM $track_exercises e
- INNER JOIN $track_attempt a
- ON (
- a.exe_id = e.exe_id AND
- e.c_id = a.c_id AND
- e.session_id = a.session_id
- )
- INNER JOIN $courseTable c
- ON c.id = a.c_id
- $courseCondition
- WHERE
- exe_exo_id = $exercise_id AND
- a.c_id = $courseId AND
- e.session_id = $session_id AND
- $answer_condition
- question_id = $question_id AND
- e.status = ''
- $courseConditionWhere
- ";
- $result = Database::query($sql);
- $return = 0;
- if ($result) {
- $good_answers = 0;
- switch ($question_type) {
- case FILL_IN_BLANKS:
- while ($row = Database::fetch_array($result, 'ASSOC')) {
- $fill_blank = self::check_fill_in_blanks(
- $correct_answer,
- $row['answer'],
- $current_answer
- );
- if (isset($fill_blank[$current_answer]) && $fill_blank[$current_answer] == 1) {
- $good_answers++;
- }
- }
- return $good_answers;
- break;
- case MATCHING:
- case MATCHING_DRAGGABLE:
- default:
- $return = Database::num_rows($result);
- }
- }
- return $return;
- }
- /**
- * @param array $answer
- * @param string $user_answer
- *
- * @return array
- */
- public static function check_fill_in_blanks($answer, $user_answer, $current_answer)
- {
- // the question is encoded like this
- // [A] B [C] D [E] F::10,10,10@1
- // number 1 before the "@" means that is a switchable fill in blank question
- // [A] B [C] D [E] F::10,10,10@ or [A] B [C] D [E] F::10,10,10
- // means that is a normal fill blank question
- // first we explode the "::"
- $pre_array = explode('::', $answer);
- // is switchable fill blank or not
- $last = count($pre_array) - 1;
- $is_set_switchable = explode('@', $pre_array[$last]);
- $switchable_answer_set = false;
- if (isset($is_set_switchable[1]) && $is_set_switchable[1] == 1) {
- $switchable_answer_set = true;
- }
- $answer = '';
- for ($k = 0; $k < $last; $k++) {
- $answer .= $pre_array[$k];
- }
- // splits weightings that are joined with a comma
- $answerWeighting = explode(',', $is_set_switchable[0]);
- // we save the answer because it will be modified
- //$temp = $answer;
- $temp = $answer;
- $answer = '';
- $j = 0;
- //initialise answer tags
- $user_tags = $correct_tags = $real_text = [];
- // the loop will stop at the end of the text
- while (1) {
- // quits the loop if there are no more blanks (detect '[')
- if (($pos = api_strpos($temp, '[')) === false) {
- // adds the end of the text
- $answer = $temp;
- $real_text[] = $answer;
- break; //no more "blanks", quit the loop
- }
- // adds the piece of text that is before the blank
- //and ends with '[' into a general storage array
- $real_text[] = api_substr($temp, 0, $pos + 1);
- $answer .= api_substr($temp, 0, $pos + 1);
- //take the string remaining (after the last "[" we found)
- $temp = api_substr($temp, $pos + 1);
- // quit the loop if there are no more blanks, and update $pos to the position of next ']'
- if (($pos = api_strpos($temp, ']')) === false) {
- // adds the end of the text
- $answer .= $temp;
- break;
- }
- $str = $user_answer;
- preg_match_all('#\[([^[]*)\]#', $str, $arr);
- $str = str_replace('\r\n', '', $str);
- $choices = $arr[1];
- $choice = [];
- $check = false;
- $i = 0;
- foreach ($choices as $item) {
- if ($current_answer === $item) {
- $check = true;
- }
- if ($check) {
- $choice[] = $item;
- $i++;
- }
- if ($i == 3) {
- break;
- }
- }
- $tmp = api_strrpos($choice[$j], ' / ');
- if ($tmp !== false) {
- $choice[$j] = api_substr($choice[$j], 0, $tmp);
- }
- $choice[$j] = trim($choice[$j]);
- //Needed to let characters ' and " to work as part of an answer
- $choice[$j] = stripslashes($choice[$j]);
- $user_tags[] = api_strtolower($choice[$j]);
- //put the contents of the [] answer tag into correct_tags[]
- $correct_tags[] = api_strtolower(api_substr($temp, 0, $pos));
- $j++;
- $temp = api_substr($temp, $pos + 1);
- }
- $answer = '';
- $real_correct_tags = $correct_tags;
- $chosen_list = [];
- $good_answer = [];
- for ($i = 0; $i < count($real_correct_tags); $i++) {
- if (!$switchable_answer_set) {
- //needed to parse ' and " characters
- $user_tags[$i] = stripslashes($user_tags[$i]);
- if ($correct_tags[$i] == $user_tags[$i]) {
- $good_answer[$correct_tags[$i]] = 1;
- } elseif (!empty($user_tags[$i])) {
- $good_answer[$correct_tags[$i]] = 0;
- } else {
- $good_answer[$correct_tags[$i]] = 0;
- }
- } else {
- // switchable fill in the blanks
- if (in_array($user_tags[$i], $correct_tags)) {
- $correct_tags = array_diff($correct_tags, $chosen_list);
- $good_answer[$correct_tags[$i]] = 1;
- } elseif (!empty($user_tags[$i])) {
- $good_answer[$correct_tags[$i]] = 0;
- } else {
- $good_answer[$correct_tags[$i]] = 0;
- }
- }
- // adds the correct word, followed by ] to close the blank
- $answer .= ' / <font color="green"><b>'.$real_correct_tags[$i].'</b></font>]';
- if (isset($real_text[$i + 1])) {
- $answer .= $real_text[$i + 1];
- }
- }
- return $good_answer;
- }
- /**
- * @param int $exercise_id
- * @param string $course_code
- * @param int $session_id
- *
- * @return int
- */
- public static function get_number_students_finish_exercise(
- $exercise_id,
- $course_code,
- $session_id
- ) {
- $track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $exercise_id = (int) $exercise_id;
- $course_code = Database::escape_string($course_code);
- $session_id = (int) $session_id;
- $sql = "SELECT DISTINCT exe_user_id
- FROM $track_exercises e
- INNER JOIN $track_attempt a
- ON (a.exe_id = e.exe_id)
- WHERE
- exe_exo_id = $exercise_id AND
- course_code = '$course_code' AND
- e.session_id = $session_id AND
- status = ''";
- $result = Database::query($sql);
- $return = 0;
- if ($result) {
- $return = Database::num_rows($result);
- }
- return $return;
- }
- /**
- * Return an HTML select menu with the student groups.
- *
- * @param string $name is the name and the id of the <select>
- * @param string $default default value for option
- * @param string $onchange
- *
- * @return string the html code of the <select>
- */
- public static function displayGroupMenu($name, $default, $onchange = "")
- {
- // check the default value of option
- $tabSelected = [$default => " selected='selected' "];
- $res = "";
- $res .= "<select name='$name' id='$name' onchange='".$onchange."' >";
- $res .= "<option value='-1'".$tabSelected["-1"].">-- ".get_lang(
- 'AllGroups'
- )." --</option>";
- $res .= "<option value='0'".$tabSelected["0"].">- ".get_lang(
- 'NotInAGroup'
- )." -</option>";
- $tabGroups = GroupManager::get_group_list();
- $currentCatId = 0;
- $countGroups = count($tabGroups);
- for ($i = 0; $i < $countGroups; $i++) {
- $tabCategory = GroupManager::get_category_from_group(
- $tabGroups[$i]['iid']
- );
- if ($tabCategory["id"] != $currentCatId) {
- $res .= "<option value='-1' disabled='disabled'>".$tabCategory["title"]."</option>";
- $currentCatId = $tabCategory["id"];
- }
- $res .= "<option ".$tabSelected[$tabGroups[$i]["id"]]."style='margin-left:40px' value='".
- $tabGroups[$i]["id"]."'>".
- $tabGroups[$i]["name"].
- "</option>";
- }
- $res .= "</select>";
- return $res;
- }
- /**
- * @param int $exe_id
- */
- public static function create_chat_exercise_session($exe_id)
- {
- if (!isset($_SESSION['current_exercises'])) {
- $_SESSION['current_exercises'] = [];
- }
- $_SESSION['current_exercises'][$exe_id] = true;
- }
- /**
- * @param int $exe_id
- */
- public static function delete_chat_exercise_session($exe_id)
- {
- if (isset($_SESSION['current_exercises'])) {
- $_SESSION['current_exercises'][$exe_id] = false;
- }
- }
- /**
- * Display the exercise results.
- *
- * @param Exercise $objExercise
- * @param int $exeId
- * @param bool $save_user_result save users results (true) or just show the results (false)
- * @param string $remainingMessage
- */
- public static function displayQuestionListByAttempt(
- $objExercise,
- $exeId,
- $save_user_result = false,
- $remainingMessage = ''
- ) {
- $origin = api_get_origin();
- $courseCode = api_get_course_id();
- $sessionId = api_get_session_id();
- // Getting attempt info
- $exercise_stat_info = $objExercise->get_stat_track_exercise_info_by_exe_id($exeId);
- // Getting question list
- $question_list = [];
- $studentInfo = [];
- if (!empty($exercise_stat_info['data_tracking'])) {
- $studentInfo = api_get_user_info($exercise_stat_info['exe_user_id']);
- $question_list = explode(',', $exercise_stat_info['data_tracking']);
- } else {
- // Try getting the question list only if save result is off
- if ($save_user_result == false) {
- $question_list = $objExercise->get_validated_question_list();
- }
- if (in_array(
- $objExercise->getFeedbackType(),
- [EXERCISE_FEEDBACK_TYPE_DIRECT, EXERCISE_FEEDBACK_TYPE_POPUP]
- )) {
- $question_list = $objExercise->get_validated_question_list();
- }
- }
- if ($objExercise->getResultAccess()) {
- if ($objExercise->hasResultsAccess($exercise_stat_info) === false) {
- echo Display::return_message(
- sprintf(get_lang('YouPassedTheLimitOfXMinutesToSeeTheResults'), $objExercise->getResultsAccess())
- );
- return false;
- }
- if (!empty($objExercise->getResultAccess())) {
- $url = api_get_path(WEB_CODE_PATH).'exercise/overview.php?'.api_get_cidreq().'&exerciseId='.$objExercise->id;
- echo $objExercise->returnTimeLeftDiv();
- echo $objExercise->showSimpleTimeControl(
- $objExercise->getResultAccessTimeDiff($exercise_stat_info),
- $url
- );
- }
- }
- $counter = 1;
- $total_score = $total_weight = 0;
- $exercise_content = null;
- // Hide results
- $show_results = false;
- $show_only_score = false;
- if (in_array($objExercise->results_disabled,
- [
- RESULT_DISABLE_SHOW_ONLY_IN_CORRECT_ANSWER,
- RESULT_DISABLE_SHOW_SCORE_AND_EXPECTED_ANSWERS,
- RESULT_DISABLE_SHOW_SCORE_AND_EXPECTED_ANSWERS_AND_RANKING,
- ]
- )) {
- $show_results = true;
- }
- if (in_array(
- $objExercise->results_disabled,
- [
- RESULT_DISABLE_SHOW_SCORE_ONLY,
- RESULT_DISABLE_SHOW_FINAL_SCORE_ONLY_WITH_CATEGORIES,
- RESULT_DISABLE_RANKING,
- ]
- )
- ) {
- $show_only_score = true;
- }
- // Not display expected answer, but score, and feedback
- $show_all_but_expected_answer = false;
- if ($objExercise->results_disabled == RESULT_DISABLE_SHOW_SCORE_ONLY &&
- $objExercise->getFeedbackType() == EXERCISE_FEEDBACK_TYPE_END
- ) {
- $show_all_but_expected_answer = true;
- $show_results = true;
- $show_only_score = false;
- }
- $showTotalScoreAndUserChoicesInLastAttempt = true;
- $showTotalScore = true;
- $showQuestionScore = true;
- if (in_array(
- $objExercise->results_disabled,
- [
- RESULT_DISABLE_SHOW_SCORE_ATTEMPT_SHOW_ANSWERS_LAST_ATTEMPT,
- RESULT_DISABLE_DONT_SHOW_SCORE_ONLY_IF_USER_FINISHES_ATTEMPTS_SHOW_ALWAYS_FEEDBACK,
- ])
- ) {
- $show_only_score = true;
- $show_results = true;
- $numberAttempts = 0;
- if ($objExercise->attempts > 0) {
- $attempts = Event::getExerciseResultsByUser(
- api_get_user_id(),
- $objExercise->id,
- api_get_course_int_id(),
- api_get_session_id(),
- $exercise_stat_info['orig_lp_id'],
- $exercise_stat_info['orig_lp_item_id'],
- 'desc'
- );
- if ($attempts) {
- $numberAttempts = count($attempts);
- }
- if ($save_user_result) {
- $numberAttempts++;
- }
- $showTotalScore = false;
- $showTotalScoreAndUserChoicesInLastAttempt = false;
- if ($numberAttempts >= $objExercise->attempts) {
- $showTotalScore = true;
- $show_results = true;
- $show_only_score = false;
- $showTotalScoreAndUserChoicesInLastAttempt = true;
- }
- }
- if ($objExercise->results_disabled ==
- RESULT_DISABLE_DONT_SHOW_SCORE_ONLY_IF_USER_FINISHES_ATTEMPTS_SHOW_ALWAYS_FEEDBACK
- ) {
- $show_only_score = false;
- $show_results = true;
- $show_all_but_expected_answer = false;
- $showTotalScore = false;
- $showQuestionScore = false;
- if ($numberAttempts >= $objExercise->attempts) {
- $showTotalScore = true;
- $showQuestionScore = true;
- }
- }
- }
- if (($show_results || $show_only_score) && $origin !== 'embeddable') {
- if (isset($exercise_stat_info['exe_user_id'])) {
- if (!empty($studentInfo)) {
- // Shows exercise header
- echo $objExercise->showExerciseResultHeader(
- $studentInfo,
- $exercise_stat_info
- );
- }
- }
- }
- // Display text when test is finished #4074 and for LP #4227
- $endOfMessage = $objExercise->getTextWhenFinished();
- if (!empty($endOfMessage)) {
- echo Display::return_message($endOfMessage, 'normal', false);
- echo "<div class='clear'> </div>";
- }
- $question_list_answers = [];
- $media_list = [];
- $category_list = [];
- $loadChoiceFromSession = false;
- $fromDatabase = true;
- $exerciseResult = null;
- $exerciseResultCoordinates = null;
- $delineationResults = null;
- if (in_array(
- $objExercise->getFeedbackType(),
- [EXERCISE_FEEDBACK_TYPE_DIRECT, EXERCISE_FEEDBACK_TYPE_POPUP]
- )) {
- $loadChoiceFromSession = true;
- $fromDatabase = false;
- $exerciseResult = Session::read('exerciseResult');
- $exerciseResultCoordinates = Session::read('exerciseResultCoordinates');
- $delineationResults = Session::read('hotspot_delineation_result');
- $delineationResults = isset($delineationResults[$objExercise->id]) ? $delineationResults[$objExercise->id] : null;
- }
- $countPendingQuestions = 0;
- $result = [];
- // Loop over all question to show results for each of them, one by one
- if (!empty($question_list)) {
- foreach ($question_list as $questionId) {
- // Creates a temporary Question object
- $objQuestionTmp = Question::read($questionId, $objExercise->course);
- // This variable came from exercise_submit_modal.php
- ob_start();
- $choice = null;
- $delineationChoice = null;
- if ($loadChoiceFromSession) {
- $choice = isset($exerciseResult[$questionId]) ? $exerciseResult[$questionId] : null;
- $delineationChoice = isset($delineationResults[$questionId]) ? $delineationResults[$questionId] : null;
- }
- // We're inside *one* question. Go through each possible answer for this question
- $result = $objExercise->manage_answer(
- $exeId,
- $questionId,
- $choice,
- 'exercise_result',
- $exerciseResultCoordinates,
- $save_user_result,
- $fromDatabase,
- $show_results,
- $objExercise->selectPropagateNeg(),
- $delineationChoice,
- $showTotalScoreAndUserChoicesInLastAttempt
- );
- if (empty($result)) {
- continue;
- }
- $total_score += $result['score'];
- $total_weight += $result['weight'];
- $question_list_answers[] = [
- 'question' => $result['open_question'],
- 'answer' => $result['open_answer'],
- 'answer_type' => $result['answer_type'],
- 'generated_oral_file' => $result['generated_oral_file'],
- ];
- $my_total_score = $result['score'];
- $my_total_weight = $result['weight'];
- // Category report
- $category_was_added_for_this_test = false;
- if (isset($objQuestionTmp->category) && !empty($objQuestionTmp->category)) {
- if (!isset($category_list[$objQuestionTmp->category]['score'])) {
- $category_list[$objQuestionTmp->category]['score'] = 0;
- }
- if (!isset($category_list[$objQuestionTmp->category]['total'])) {
- $category_list[$objQuestionTmp->category]['total'] = 0;
- }
- $category_list[$objQuestionTmp->category]['score'] += $my_total_score;
- $category_list[$objQuestionTmp->category]['total'] += $my_total_weight;
- $category_was_added_for_this_test = true;
- }
- if (isset($objQuestionTmp->category_list) && !empty($objQuestionTmp->category_list)) {
- foreach ($objQuestionTmp->category_list as $category_id) {
- $category_list[$category_id]['score'] += $my_total_score;
- $category_list[$category_id]['total'] += $my_total_weight;
- $category_was_added_for_this_test = true;
- }
- }
- // No category for this question!
- if ($category_was_added_for_this_test == false) {
- if (!isset($category_list['none']['score'])) {
- $category_list['none']['score'] = 0;
- }
- if (!isset($category_list['none']['total'])) {
- $category_list['none']['total'] = 0;
- }
- $category_list['none']['score'] += $my_total_score;
- $category_list['none']['total'] += $my_total_weight;
- }
- if ($objExercise->selectPropagateNeg() == 0 && $my_total_score < 0) {
- $my_total_score = 0;
- }
- $comnt = null;
- if ($show_results) {
- $comnt = Event::get_comments($exeId, $questionId);
- $teacherAudio = self::getOralFeedbackAudio(
- $exeId,
- $questionId,
- api_get_user_id()
- );
- if (!empty($comnt) || $teacherAudio) {
- echo '<b>'.get_lang('Feedback').'</b>';
- }
- if (!empty($comnt)) {
- echo self::getFeedbackText($comnt);
- }
- if ($teacherAudio) {
- echo $teacherAudio;
- }
- }
- $score = [];
- if ($show_results) {
- $scorePassed = $my_total_score >= $my_total_weight;
- if (function_exists('bccomp')) {
- $compareResult = bccomp($my_total_score, $my_total_weight, 3);
- $scorePassed = $compareResult === 1 || $compareResult === 0;
- }
- $score = [
- 'result' => self::show_score(
- $my_total_score,
- $my_total_weight,
- false
- ),
- 'pass' => $scorePassed,
- 'score' => $my_total_score,
- 'weight' => $my_total_weight,
- 'comments' => $comnt,
- 'user_answered' => $result['user_answered'],
- ];
- }
- if (in_array($objQuestionTmp->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION])) {
- $reviewScore = [
- 'score' => $my_total_score,
- 'comments' => Event::get_comments($exeId, $questionId),
- ];
- $check = $objQuestionTmp->isQuestionWaitingReview($reviewScore);
- if ($check === false) {
- $countPendingQuestions++;
- }
- }
- $contents = ob_get_clean();
- $question_content = '';
- if ($show_results) {
- $question_content = '<div class="question_row_answer">';
- if ($showQuestionScore == false) {
- $score = [];
- }
- // Shows question title an description
- $question_content .= $objQuestionTmp->return_header(
- $objExercise,
- $counter,
- $score
- );
- }
- $counter++;
- $question_content .= $contents;
- if ($show_results) {
- $question_content .= '</div>';
- }
- if ($objExercise->showExpectedChoice()) {
- $exercise_content .= Display::div(
- Display::panel($question_content),
- ['class' => 'question-panel']
- );
- } else {
- // $show_all_but_expected_answer should not happen at
- // the same time as $show_results
- if ($show_results && !$show_only_score) {
- $exercise_content .= Display::div(
- Display::panel($question_content),
- ['class' => 'question-panel']
- );
- }
- }
- } // end foreach() block that loops over all questions
- }
- $totalScoreText = null;
- $certificateBlock = '';
- if (($show_results || $show_only_score) && $showTotalScore) {
- if ($result['answer_type'] == MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY) {
- echo '<h1 style="text-align : center; margin : 20px 0;">'.get_lang('YourResults').'</h1><br />';
- }
- $totalScoreText .= '<div class="question_row_score">';
- if ($result['answer_type'] == MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY) {
- $totalScoreText .= self::getQuestionDiagnosisRibbon(
- $objExercise,
- $total_score,
- $total_weight,
- true
- );
- } else {
- $pluginEvaluation = QuestionOptionsEvaluationPlugin::create();
- if ('true' === $pluginEvaluation->get(QuestionOptionsEvaluationPlugin::SETTING_ENABLE)) {
- $formula = $pluginEvaluation->getFormulaForExercise($objExercise->selectId());
- if (!empty($formula)) {
- $total_score = $pluginEvaluation->getResultWithFormula($exeId, $formula);
- $total_weight = $pluginEvaluation->getMaxScore();
- }
- }
- $totalScoreText .= self::getTotalScoreRibbon(
- $objExercise,
- $total_score,
- $total_weight,
- true,
- $countPendingQuestions
- );
- }
- $totalScoreText .= '</div>';
- if (!empty($studentInfo)) {
- $certificateBlock = self::generateAndShowCertificateBlock(
- $total_score,
- $total_weight,
- $objExercise,
- $studentInfo['id'],
- $courseCode,
- $sessionId
- );
- }
- }
- if ($result['answer_type'] == MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY) {
- $chartMultiAnswer = MultipleAnswerTrueFalseDegreeCertainty::displayStudentsChartResults(
- $exeId,
- $objExercise
- );
- echo $chartMultiAnswer;
- }
- if (!empty($category_list) && ($show_results || $show_only_score)) {
- // Adding total
- $category_list['total'] = [
- 'score' => $total_score,
- 'total' => $total_weight,
- ];
- echo TestCategory::get_stats_table_by_attempt(
- $objExercise->id,
- $category_list
- );
- }
- if ($show_all_but_expected_answer) {
- $exercise_content .= Display::return_message(get_lang('ExerciseWithFeedbackWithoutCorrectionComment'));
- }
- // Remove audio auto play from questions on results page - refs BT#7939
- $exercise_content = preg_replace(
- ['/autoplay[\=\".+\"]+/', '/autostart[\=\".+\"]+/'],
- '',
- $exercise_content
- );
- echo $totalScoreText;
- echo $certificateBlock;
- // Ofaj change BT#11784
- if (api_get_configuration_value('quiz_show_description_on_results_page') &&
- !empty($objExercise->description)
- ) {
- echo Display::div($objExercise->description, ['class' => 'exercise_description']);
- }
- echo $exercise_content;
- if (!$show_only_score) {
- echo $totalScoreText;
- }
- if ($save_user_result) {
- // Tracking of results
- if ($exercise_stat_info) {
- $learnpath_id = $exercise_stat_info['orig_lp_id'];
- $learnpath_item_id = $exercise_stat_info['orig_lp_item_id'];
- $learnpath_item_view_id = $exercise_stat_info['orig_lp_item_view_id'];
- if (api_is_allowed_to_session_edit()) {
- Event::updateEventExercise(
- $exercise_stat_info['exe_id'],
- $objExercise->selectId(),
- $total_score,
- $total_weight,
- api_get_session_id(),
- $learnpath_id,
- $learnpath_item_id,
- $learnpath_item_view_id,
- $exercise_stat_info['exe_duration'],
- $question_list
- );
- $allowStats = api_get_configuration_value('allow_gradebook_stats');
- if ($allowStats) {
- $objExercise->generateStats(
- $objExercise->selectId(),
- api_get_course_info(),
- api_get_session_id()
- );
- }
- }
- }
- // Send notification at the end
- if (!api_is_allowed_to_edit(null, true) &&
- !api_is_excluded_user_type()
- ) {
- $objExercise->send_mail_notification_for_exam(
- 'end',
- $question_list_answers,
- $origin,
- $exeId,
- $total_score,
- $total_weight
- );
- }
- }
- if (in_array(
- $objExercise->selectResultsDisabled(),
- [RESULT_DISABLE_RANKING, RESULT_DISABLE_SHOW_SCORE_AND_EXPECTED_ANSWERS_AND_RANKING]
- )) {
- echo Display::page_header(get_lang('Ranking'), null, 'h4');
- echo self::displayResultsInRanking(
- $objExercise->iId,
- api_get_user_id(),
- api_get_course_int_id(),
- api_get_session_id()
- );
- }
- if (!empty($remainingMessage)) {
- echo Display::return_message($remainingMessage, 'normal', false);
- }
- }
- /**
- * Display the ranking of results in a exercise.
- *
- * @param int $exerciseId
- * @param int $currentUserId
- * @param int $courseId
- * @param int $sessionId
- *
- * @return string
- */
- public static function displayResultsInRanking($exerciseId, $currentUserId, $courseId, $sessionId = 0)
- {
- $data = self::exerciseResultsInRanking($exerciseId, $courseId, $sessionId);
- $table = new HTML_Table(['class' => 'table table-hover table-bordered']);
- $table->setHeaderContents(0, 0, get_lang('Position'), ['class' => 'text-right']);
- $table->setHeaderContents(0, 1, get_lang('Username'));
- $table->setHeaderContents(0, 2, get_lang('Score'), ['class' => 'text-right']);
- $table->setHeaderContents(0, 3, get_lang('Date'), ['class' => 'text-center']);
- foreach ($data as $r => $item) {
- if (!isset($item[1])) {
- continue;
- }
- $selected = $item[1]->getId() == $currentUserId;
- foreach ($item as $c => $value) {
- $table->setCellContents($r + 1, $c, $value);
- $attrClass = '';
- if (in_array($c, [0, 2])) {
- $attrClass = 'text-right';
- } elseif (3 == $c) {
- $attrClass = 'text-center';
- }
- if ($selected) {
- $attrClass .= ' warning';
- }
- $table->setCellAttributes($r + 1, $c, ['class' => $attrClass]);
- }
- }
- return $table->toHtml();
- }
- /**
- * Get the ranking for results in a exercise.
- * Function used internally by ExerciseLib::displayResultsInRanking.
- *
- * @param int $exerciseId
- * @param int $courseId
- * @param int $sessionId
- *
- * @return array
- */
- public static function exerciseResultsInRanking($exerciseId, $courseId, $sessionId = 0)
- {
- $em = Database::getManager();
- $dql = 'SELECT DISTINCT te.exeUserId FROM ChamiloCoreBundle:TrackEExercises te WHERE te.exeExoId = :id AND te.cId = :cId';
- $dql .= api_get_session_condition($sessionId, true, false, 'te.sessionId');
- $result = $em
- ->createQuery($dql)
- ->setParameters(['id' => $exerciseId, 'cId' => $courseId])
- ->getScalarResult();
- $data = [];
- /** @var TrackEExercises $item */
- foreach ($result as $item) {
- $bestAttemp = self::get_best_attempt_by_user($item['exeUserId'], $exerciseId, $courseId, $sessionId = 0);
- $data[] = $bestAttemp;
- }
- usort(
- $data,
- function ($a, $b) {
- if ($a['exe_result'] != $b['exe_result']) {
- return $a['exe_result'] > $b['exe_result'] ? -1 : 1;
- }
- if ($a['exe_date'] != $b['exe_date']) {
- return $a['exe_date'] < $b['exe_date'] ? -1 : 1;
- }
- return 0;
- }
- );
- // flags to display the same position in case of tie
- $lastScore = $data[0]['exe_result'];
- $position = 1;
- $data = array_map(
- function ($item) use (&$lastScore, &$position) {
- if ($item['exe_result'] < $lastScore) {
- $position++;
- }
- $lastScore = $item['exe_result'];
- return [
- $position,
- api_get_user_entity($item['exe_user_id']),
- self::show_score($item['exe_result'], $item['exe_weighting'], true, true, true),
- api_convert_and_format_date($item['exe_date'], DATE_TIME_FORMAT_SHORT),
- ];
- },
- $data
- );
- return $data;
- }
- /**
- * Get a special ribbon on top of "degree of certainty" questions (
- * variation from getTotalScoreRibbon() for other question types).
- *
- * @param Exercise $objExercise
- * @param float $score
- * @param float $weight
- * @param bool $checkPassPercentage
- *
- * @return string
- */
- public static function getQuestionDiagnosisRibbon($objExercise, $score, $weight, $checkPassPercentage = false)
- {
- $displayChartDegree = true;
- $ribbon = $displayChartDegree ? '<div class="ribbon">' : '';
- if ($checkPassPercentage) {
- $isSuccess = self::isSuccessExerciseResult(
- $score, $weight, $objExercise->selectPassPercentage()
- );
- // Color the final test score if pass_percentage activated
- $ribbonTotalSuccessOrError = '';
- if (self::isPassPercentageEnabled($objExercise->selectPassPercentage())) {
- if ($isSuccess) {
- $ribbonTotalSuccessOrError = ' ribbon-total-success';
- } else {
- $ribbonTotalSuccessOrError = ' ribbon-total-error';
- }
- }
- $ribbon .= $displayChartDegree ? '<div class="rib rib-total '.$ribbonTotalSuccessOrError.'">' : '';
- } else {
- $ribbon .= $displayChartDegree ? '<div class="rib rib-total">' : '';
- }
- if ($displayChartDegree) {
- $ribbon .= '<h3>'.get_lang('YourTotalScore').': ';
- $ribbon .= self::show_score($score, $weight, false, true);
- $ribbon .= '</h3>';
- $ribbon .= '</div>';
- }
- if ($checkPassPercentage) {
- $ribbon .= self::showSuccessMessage(
- $score,
- $weight,
- $objExercise->selectPassPercentage()
- );
- }
- $ribbon .= $displayChartDegree ? '</div>' : '';
- return $ribbon;
- }
- /**
- * @param Exercise $objExercise
- * @param float $score
- * @param float $weight
- * @param bool $checkPassPercentage
- * @param int $countPendingQuestions
- *
- * @return string
- */
- public static function getTotalScoreRibbon(
- Exercise $objExercise,
- $score,
- $weight,
- $checkPassPercentage = false,
- $countPendingQuestions = 0
- ) {
- $hide = (int) $objExercise->getPageConfigurationAttribute('hide_total_score');
- if ($hide === 1) {
- return '';
- }
- $passPercentage = $objExercise->selectPassPercentage();
- $ribbon = '<div class="title-score">';
- if ($checkPassPercentage) {
- $isSuccess = self::isSuccessExerciseResult(
- $score,
- $weight,
- $passPercentage
- );
- // Color the final test score if pass_percentage activated
- $class = '';
- if (self::isPassPercentageEnabled($passPercentage)) {
- if ($isSuccess) {
- $class = ' ribbon-total-success';
- } else {
- $class = ' ribbon-total-error';
- }
- }
- $ribbon .= '<div class="total '.$class.'">';
- } else {
- $ribbon .= '<div class="total">';
- }
- $ribbon .= '<h3>'.get_lang('YourTotalScore').': ';
- $ribbon .= self::show_score($score, $weight, false, true);
- $ribbon .= '</h3>';
- $ribbon .= '</div>';
- if ($checkPassPercentage) {
- $ribbon .= self::showSuccessMessage(
- $score,
- $weight,
- $passPercentage
- );
- }
- $ribbon .= '</div>';
- if (!empty($countPendingQuestions)) {
- $ribbon .= '<br />';
- $ribbon .= Display::return_message(
- sprintf(
- get_lang('TempScoreXQuestionsNotCorrectedYet'),
- $countPendingQuestions
- ),
- 'warning'
- );
- }
- return $ribbon;
- }
- /**
- * @param int $countLetter
- *
- * @return mixed
- */
- public static function detectInputAppropriateClass($countLetter)
- {
- $limits = [
- 0 => 'input-mini',
- 10 => 'input-mini',
- 15 => 'input-medium',
- 20 => 'input-xlarge',
- 40 => 'input-xlarge',
- 60 => 'input-xxlarge',
- 100 => 'input-xxlarge',
- 200 => 'input-xxlarge',
- ];
- foreach ($limits as $size => $item) {
- if ($countLetter <= $size) {
- return $item;
- }
- }
- return $limits[0];
- }
- /**
- * @param int $senderId
- * @param array $course_info
- * @param string $test
- * @param string $url
- *
- * @return string
- */
- public static function getEmailNotification($senderId, $course_info, $test, $url)
- {
- $teacher_info = api_get_user_info($senderId);
- $from_name = api_get_person_name(
- $teacher_info['firstname'],
- $teacher_info['lastname'],
- null,
- PERSON_NAME_EMAIL_ADDRESS
- );
- $view = new Template('', false, false, false, false, false, false);
- $view->assign('course_title', Security::remove_XSS($course_info['name']));
- $view->assign('test_title', Security::remove_XSS($test));
- $view->assign('url', $url);
- $view->assign('teacher_name', $from_name);
- $template = $view->get_template('mail/exercise_result_alert_body.tpl');
- return $view->fetch($template);
- }
- /**
- * @return string
- */
- public static function getNotCorrectedYetText()
- {
- return Display::return_message(get_lang('notCorrectedYet'), 'warning');
- }
- /**
- * @param string $message
- *
- * @return string
- */
- public static function getFeedbackText($message)
- {
- return Display::return_message($message, 'warning', false);
- }
- /**
- * Get the recorder audio component for save a teacher audio feedback.
- *
- * @param int $attemptId
- * @param int $questionId
- * @param int $userId
- *
- * @return string
- */
- public static function getOralFeedbackForm($attemptId, $questionId, $userId)
- {
- $view = new Template('', false, false, false, false, false, false);
- $view->assign('user_id', $userId);
- $view->assign('question_id', $questionId);
- $view->assign('directory', "/../exercises/teacher_audio/$attemptId/");
- $view->assign('file_name', "{$questionId}_{$userId}");
- $template = $view->get_template('exercise/oral_expression.tpl');
- return $view->fetch($template);
- }
- /**
- * Get the audio componen for a teacher audio feedback.
- *
- * @param int $attemptId
- * @param int $questionId
- * @param int $userId
- *
- * @return string
- */
- public static function getOralFeedbackAudio($attemptId, $questionId, $userId)
- {
- $courseInfo = api_get_course_info();
- $sessionId = api_get_session_id();
- $groupId = api_get_group_id();
- $sysCourseDir = api_get_path(SYS_COURSE_PATH).$courseInfo['path'];
- $webCourseDir = api_get_path(WEB_COURSE_PATH).$courseInfo['path'];
- $fileName = "{$questionId}_{$userId}".DocumentManager::getDocumentSuffix($courseInfo, $sessionId, $groupId);
- $filePath = null;
- $relFilePath = "/exercises/teacher_audio/$attemptId/$fileName";
- if (file_exists($sysCourseDir.$relFilePath.'.ogg')) {
- $filePath = $webCourseDir.$relFilePath.'.ogg';
- } elseif (file_exists($sysCourseDir.$relFilePath.'.wav.wav')) {
- $filePath = $webCourseDir.$relFilePath.'.wav.wav';
- } elseif (file_exists($sysCourseDir.$relFilePath.'.wav')) {
- $filePath = $webCourseDir.$relFilePath.'.wav';
- }
- if (!$filePath) {
- return '';
- }
- return Display::tag(
- 'audio',
- null,
- ['src' => $filePath]
- );
- }
- /**
- * @return array
- */
- public static function getNotificationSettings()
- {
- $emailAlerts = [
- 2 => get_lang('SendEmailToTeacherWhenStudentStartQuiz'),
- 1 => get_lang('SendEmailToTeacherWhenStudentEndQuiz'), // default
- 3 => get_lang('SendEmailToTeacherWhenStudentEndQuizOnlyIfOpenQuestion'),
- 4 => get_lang('SendEmailToTeacherWhenStudentEndQuizOnlyIfOralQuestion'),
- ];
- return $emailAlerts;
- }
- /**
- * Get the additional actions added in exercise_additional_teacher_modify_actions configuration.
- *
- * @param int $exerciseId
- * @param int $iconSize
- *
- * @return string
- */
- public static function getAdditionalTeacherActions($exerciseId, $iconSize = ICON_SIZE_SMALL)
- {
- $additionalActions = api_get_configuration_value('exercise_additional_teacher_modify_actions') ?: [];
- $actions = [];
- foreach ($additionalActions as $additionalAction) {
- $actions[] = call_user_func(
- $additionalAction,
- $exerciseId,
- $iconSize
- );
- }
- return implode(PHP_EOL, $actions);
- }
- /**
- * @param DateTime $time
- * @param int $userId
- * @param int $courseId
- * @param int $sessionId
- *
- * @throws \Doctrine\ORM\Query\QueryException
- *
- * @return int
- */
- public static function countAnsweredQuestionsByUserAfterTime(DateTime $time, $userId, $courseId, $sessionId)
- {
- $em = Database::getManager();
- $time = api_get_utc_datetime($time->format('Y-m-d H:i:s'), false, true);
- $result = $em
- ->createQuery('
- SELECT COUNT(ea) FROM ChamiloCoreBundle:TrackEAttempt ea
- WHERE ea.userId = :user AND ea.cId = :course AND ea.sessionId = :session
- AND ea.tms > :time
- ')
- ->setParameters(['user' => $userId, 'course' => $courseId, 'session' => $sessionId, 'time' => $time])
- ->getSingleScalarResult();
- return $result;
- }
- /**
- * @param int $userId
- * @param int $numberOfQuestions
- * @param int $courseId
- * @param int $sessionId
- *
- * @throws \Doctrine\ORM\Query\QueryException
- *
- * @return bool
- */
- public static function isQuestionsLimitPerDayReached($userId, $numberOfQuestions, $courseId, $sessionId)
- {
- $questionsLimitPerDay = (int) api_get_course_setting('quiz_question_limit_per_day');
- if ($questionsLimitPerDay <= 0) {
- return false;
- }
- $midnightTime = ChamiloApi::getServerMidnightTime();
- $answeredQuestionsCount = self::countAnsweredQuestionsByUserAfterTime(
- $midnightTime,
- $userId,
- $courseId,
- $sessionId
- );
- return ($answeredQuestionsCount + $numberOfQuestions) > $questionsLimitPerDay;
- }
- /**
- * Check if an exercise complies with the requirements to be embedded in the mobile app or a video.
- * By making sure it is set on one question per page and it only contains unique-answer or multiple-answer questions
- * or unique-answer image. And that the exam does not have immediate feedback.
- *
- * @param array $exercise Exercise info
- *
- * @throws \Doctrine\ORM\Query\QueryException
- *
- * @return bool
- */
- public static function isQuizEmbeddable(array $exercise)
- {
- $em = Database::getManager();
- if (ONE_PER_PAGE != $exercise['type'] ||
- in_array($exercise['feedback_type'], [EXERCISE_FEEDBACK_TYPE_DIRECT, EXERCISE_FEEDBACK_TYPE_POPUP])
- ) {
- return false;
- }
- $countAll = $em
- ->createQuery('SELECT COUNT(qq)
- FROM ChamiloCourseBundle:CQuizQuestion qq
- INNER JOIN ChamiloCourseBundle:CQuizRelQuestion qrq
- WITH qq.iid = qrq.questionId
- WHERE qrq.exerciceId = :id'
- )
- ->setParameter('id', $exercise['iid'])
- ->getSingleScalarResult();
- $countOfAllowed = $em
- ->createQuery('SELECT COUNT(qq)
- FROM ChamiloCourseBundle:CQuizQuestion qq
- INNER JOIN ChamiloCourseBundle:CQuizRelQuestion qrq
- WITH qq.iid = qrq.questionId
- WHERE qrq.exerciceId = :id AND qq.type IN (:types)'
- )
- ->setParameters(
- [
- 'id' => $exercise['iid'],
- 'types' => [UNIQUE_ANSWER, MULTIPLE_ANSWER, UNIQUE_ANSWER_IMAGE],
- ]
- )
- ->getSingleScalarResult();
- return $countAll === $countOfAllowed;
- }
- /**
- * Generate a certificate linked to current quiz and.
- * Return the HTML block with links to download and view the certificate.
- *
- * @param float $totalScore
- * @param float $totalWeight
- * @param Exercise $objExercise
- * @param int $studentId
- * @param string $courseCode
- * @param int $sessionId
- *
- * @return string
- */
- public static function generateAndShowCertificateBlock(
- $totalScore,
- $totalWeight,
- Exercise $objExercise,
- $studentId,
- $courseCode,
- $sessionId = 0
- ) {
- if (!api_get_configuration_value('quiz_generate_certificate_ending') ||
- !self::isSuccessExerciseResult($totalScore, $totalWeight, $objExercise->selectPassPercentage())
- ) {
- return '';
- }
- /** @var Category $category */
- $category = Category::load(null, null, $courseCode, null, null, $sessionId, 'ORDER By id');
- if (empty($category)) {
- return '';
- }
- /** @var Category $category */
- $category = $category[0];
- $categoryId = $category->get_id();
- $link = LinkFactory::load(
- null,
- null,
- $objExercise->selectId(),
- null,
- $courseCode,
- $categoryId
- );
- if (empty($link)) {
- return '';
- }
- $resourceDeletedMessage = $category->show_message_resource_delete($courseCode);
- if (false !== $resourceDeletedMessage || api_is_allowed_to_edit() || api_is_excluded_user_type()) {
- return '';
- }
- $certificate = Category::generateUserCertificate($categoryId, $studentId);
- if (!is_array($certificate)) {
- return '';
- }
- return Category::getDownloadCertificateBlock($certificate);
- }
- }
|