testforce-engine 0.0.3 → 1.0.0-beta-1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.mts CHANGED
@@ -1,4 +1,22 @@
1
- import React, { PropsWithChildren } from 'react';
1
+ import React from 'react';
2
+
3
+ declare enum AttemptStatus {
4
+ ATTEMPTED = "ATTEMPTED",
5
+ NOT_ATTEMPTED = "NOT_ATTEMPTED",
6
+ NOT_VISITED = "NOT_VISITED",
7
+ MARKED_FOR_REVIEW = "MARKED_FOR_REVIEW",
8
+ ATTEMPTED_AND_MARKED_FOR_REVIEW = "ATTEMPTED_AND_MARKED_FOR_REVIEW",
9
+ ACITVE_QUESTION = "ACITVE_QUESTION"
10
+ }
11
+ interface UserSettings {
12
+ selectedFilter?: AttemptStatus;
13
+ }
14
+ interface SolutionDetails$1 {
15
+ markedSolution?: string[];
16
+ markedForReview?: boolean;
17
+ timeTaken?: number;
18
+ isSaved?: boolean;
19
+ }
2
20
 
3
21
  declare enum TestType {
4
22
  QUIZ = "QUIZ",
@@ -79,10 +97,10 @@ interface Question {
79
97
  solutionDescription?: string;
80
98
  displayOrder: number;
81
99
  duration: number;
82
- questionResponse: SolutionDetails$1;
100
+ questionResponse: SolutionDetails;
83
101
  imageUrl: string;
84
102
  }
85
- interface SolutionDetails$1 {
103
+ interface SolutionDetails {
86
104
  markedSolution?: string[];
87
105
  markedForReview?: boolean;
88
106
  timeTaken?: number;
@@ -92,99 +110,82 @@ interface Option {
92
110
  _id: string;
93
111
  }
94
112
 
95
- declare enum AttemptStatus {
96
- ATTEMPTED = "ATTEMPTED",
97
- NOT_ATTEMPTED = "NOT_ATTEMPTED",
98
- NOT_VISITED = "NOT_VISITED",
99
- MARKED_FOR_REVIEW = "MARKED_FOR_REVIEW",
100
- ATTEMPTED_AND_MARKED_FOR_REVIEW = "ATTEMPTED_AND_MARKED_FOR_REVIEW",
101
- ACITVE_QUESTION = "ACITVE_QUESTION"
102
- }
103
- interface UserSettings {
104
- selectedFilter?: AttemptStatus;
105
- }
106
- interface SolutionDetails {
107
- markedSolution?: string[];
108
- markedForReview?: boolean;
109
- timeTaken?: number;
110
- isSaved?: boolean;
111
- }
112
- interface SubmitTest {
113
- payload: Record<string, any>;
114
- pause: boolean;
115
- autoSubmit: boolean;
116
- }
117
-
118
- declare const TestEngineProvider: ({ test, submitTest, ...props }: {
119
- test: Test;
120
- isSolution?: boolean;
121
- submitTest: React.Dispatch<SubmitTest>;
122
- } & PropsWithChildren) => React.JSX.Element;
123
-
124
113
  declare const useTestEngine: () => {
125
- currentSection: Section | undefined;
126
- currentQuestion: Question | undefined;
127
- handleSubmitTest: (pause?: boolean, autoSubmit?: boolean) => Promise<Record<string, any>>;
128
- handleUpdateTestDetails: (value: Test) => void;
129
- handleUpdateQuestionCoordinates: (value: {
130
- currentQuestionIndex: number;
131
- currentSectionIndex: number;
132
- }) => Promise<void>;
133
- handleUpdateSolutions: (value: Record<string, SolutionDetails>) => void;
134
- handleUpdateCurrentAnswer: (value: SolutionDetails) => void;
135
- handleUpdateUserSettings: (value: UserSettings) => void;
136
- handleResetLocal: () => Promise<void>;
137
114
  test: Test | undefined;
138
- solutions: Record<string, SolutionDetails>;
115
+ solutions: Record<string, SolutionDetails$1>;
139
116
  currentQuestionIndex: number;
140
117
  currentSectionIndex: number;
141
- currentQuestionAnswer: SolutionDetails | undefined;
118
+ currentQuestionAnswer: SolutionDetails$1 | undefined;
142
119
  userSettings: UserSettings | undefined;
143
120
  isSolution?: boolean;
144
121
  submitting?: boolean;
122
+ timer: Record<string, number>;
123
+ isHydrated: boolean;
124
+ setTest: (test: Test, shouldUpdateSolution: boolean) => void;
125
+ setQuestionCoordinates: (sectionIndex: number, questionIndex: number) => void;
126
+ setSolution: (questionId: string, details: SolutionDetails$1) => void;
127
+ setCurrentAnswer: (answer: SolutionDetails$1 | undefined) => void;
128
+ setUserSettings: (settings: UserSettings) => void;
129
+ setTestSubmitting: (submitting: boolean) => void;
130
+ incrementTimer: (quesId: string) => void;
131
+ decrementTimer: (testId: string) => void;
132
+ reset: () => void;
133
+ setIsHydrated: (isHydrated: boolean) => void;
134
+ handleSubmitTest: (pause?: boolean, autoSubmit?: boolean) => Record<string, any>;
135
+ currentSection: Section | undefined;
136
+ currentQuestion: Question | undefined;
145
137
  };
146
138
 
147
- declare const useTestEngineBody: () => {
148
- isQuizAnswer: boolean;
149
- handleClearResponse: () => void;
150
- handleSelectOption: (optionId: string) => void;
151
- handleIntegerInputChange: (val: string) => void;
152
- handleMarkForReview: () => void;
153
- };
154
-
155
- declare const useTestEngineFooter: () => {
156
- handleNext: () => void;
157
- handlePrevious: () => void;
158
- handleSaveAnswer: () => void;
159
- handleMarkForReview: () => void;
160
- isLastQuestion: boolean;
161
- isFirstQuestion: boolean;
139
+ declare const useTestTimer: () => {
140
+ timer: number;
162
141
  };
163
142
 
164
- declare const useTestQuestionStatus: ({ includeAllSections, }?: {
143
+ type QuestionStatusType = "correct" | "incorrect" | "skipped";
144
+ declare const useQuestions: ({ includeAllSections, }?: {
165
145
  includeAllSections?: boolean;
166
146
  }) => {
167
147
  getQuestionStatus: (qi: number, si: number) => AttemptStatus.ATTEMPTED | AttemptStatus.NOT_ATTEMPTED | AttemptStatus.NOT_VISITED | AttemptStatus.MARKED_FOR_REVIEW | AttemptStatus.ATTEMPTED_AND_MARKED_FOR_REVIEW;
168
148
  questionStatusMap: Record<AttemptStatus, {
169
149
  count: number;
170
150
  }>;
171
- getAnsweredQuestionStatus: (i: number, si: number) => AttemptStatus.ATTEMPTED | AttemptStatus.NOT_ATTEMPTED | AttemptStatus.NOT_VISITED;
151
+ getAnsweredQuestionStatus: (qi: number, si: number) => AttemptStatus.ATTEMPTED | AttemptStatus.NOT_ATTEMPTED | AttemptStatus.NOT_VISITED;
172
152
  areArraysEqual: (a: string[], b: string[]) => boolean;
173
- };
174
-
175
- type QuestionStatusType = "correct" | "incorrect" | "skipped";
176
- declare const useTestSolutionBody: () => {
177
- correctAnswer: string;
178
- currentAnswerMarks: string | number;
179
153
  currentQuestionStatus: QuestionStatusType;
154
+ currentAnswerMarks: string | number;
155
+ correctAnswer: string;
180
156
  };
181
157
 
182
- declare const useTestTimer: (type: "web" | "app") => {
158
+ declare const useQuestionTimer: () => {
183
159
  timer: number;
184
160
  };
185
161
 
186
- declare const useQuestionStopWatch: (type?: "web" | "app") => {
187
- questionStopWatch: number;
162
+ declare const useTestActions: () => {
163
+ isQuizAnswer: boolean;
164
+ handleClearResponse: () => void;
165
+ handleSelectOption: (optionId: string) => void;
166
+ handleIntegerInputChange: (val: string) => void;
167
+ handleMarkForReview: () => void;
168
+ handleNext: () => void;
169
+ handlePrevious: () => void;
170
+ handleSaveAnswer: () => void;
171
+ isLastQuestion: boolean;
172
+ isFirstQuestion: boolean;
188
173
  };
189
174
 
190
- export { AttemptStatus, type SubmitTest, TestEngineProvider, useQuestionStopWatch, useTestEngine, useTestEngineBody, useTestEngineFooter, useTestQuestionStatus, useTestSolutionBody, useTestTimer };
175
+ interface SubmitTest {
176
+ payload: Record<string, any>;
177
+ pause: boolean;
178
+ autoSubmit: boolean;
179
+ }
180
+
181
+ interface ProviderRef {
182
+ reset: () => void;
183
+ }
184
+ declare const Provider: React.ForwardRefExoticComponent<{
185
+ test: Test;
186
+ isSolution?: boolean;
187
+ submitTest: React.Dispatch<SubmitTest>;
188
+ children: React.ReactNode;
189
+ } & React.RefAttributes<ProviderRef>>;
190
+
191
+ export { AttemptStatus, type ProviderRef, type SubmitTest, Provider as TestEngineProvider, useQuestionTimer, useQuestions, useTestActions, useTestEngine, useTestTimer };
package/dist/index.d.ts CHANGED
@@ -1,4 +1,22 @@
1
- import React, { PropsWithChildren } from 'react';
1
+ import React from 'react';
2
+
3
+ declare enum AttemptStatus {
4
+ ATTEMPTED = "ATTEMPTED",
5
+ NOT_ATTEMPTED = "NOT_ATTEMPTED",
6
+ NOT_VISITED = "NOT_VISITED",
7
+ MARKED_FOR_REVIEW = "MARKED_FOR_REVIEW",
8
+ ATTEMPTED_AND_MARKED_FOR_REVIEW = "ATTEMPTED_AND_MARKED_FOR_REVIEW",
9
+ ACITVE_QUESTION = "ACITVE_QUESTION"
10
+ }
11
+ interface UserSettings {
12
+ selectedFilter?: AttemptStatus;
13
+ }
14
+ interface SolutionDetails$1 {
15
+ markedSolution?: string[];
16
+ markedForReview?: boolean;
17
+ timeTaken?: number;
18
+ isSaved?: boolean;
19
+ }
2
20
 
3
21
  declare enum TestType {
4
22
  QUIZ = "QUIZ",
@@ -79,10 +97,10 @@ interface Question {
79
97
  solutionDescription?: string;
80
98
  displayOrder: number;
81
99
  duration: number;
82
- questionResponse: SolutionDetails$1;
100
+ questionResponse: SolutionDetails;
83
101
  imageUrl: string;
84
102
  }
85
- interface SolutionDetails$1 {
103
+ interface SolutionDetails {
86
104
  markedSolution?: string[];
87
105
  markedForReview?: boolean;
88
106
  timeTaken?: number;
@@ -92,99 +110,82 @@ interface Option {
92
110
  _id: string;
93
111
  }
94
112
 
95
- declare enum AttemptStatus {
96
- ATTEMPTED = "ATTEMPTED",
97
- NOT_ATTEMPTED = "NOT_ATTEMPTED",
98
- NOT_VISITED = "NOT_VISITED",
99
- MARKED_FOR_REVIEW = "MARKED_FOR_REVIEW",
100
- ATTEMPTED_AND_MARKED_FOR_REVIEW = "ATTEMPTED_AND_MARKED_FOR_REVIEW",
101
- ACITVE_QUESTION = "ACITVE_QUESTION"
102
- }
103
- interface UserSettings {
104
- selectedFilter?: AttemptStatus;
105
- }
106
- interface SolutionDetails {
107
- markedSolution?: string[];
108
- markedForReview?: boolean;
109
- timeTaken?: number;
110
- isSaved?: boolean;
111
- }
112
- interface SubmitTest {
113
- payload: Record<string, any>;
114
- pause: boolean;
115
- autoSubmit: boolean;
116
- }
117
-
118
- declare const TestEngineProvider: ({ test, submitTest, ...props }: {
119
- test: Test;
120
- isSolution?: boolean;
121
- submitTest: React.Dispatch<SubmitTest>;
122
- } & PropsWithChildren) => React.JSX.Element;
123
-
124
113
  declare const useTestEngine: () => {
125
- currentSection: Section | undefined;
126
- currentQuestion: Question | undefined;
127
- handleSubmitTest: (pause?: boolean, autoSubmit?: boolean) => Promise<Record<string, any>>;
128
- handleUpdateTestDetails: (value: Test) => void;
129
- handleUpdateQuestionCoordinates: (value: {
130
- currentQuestionIndex: number;
131
- currentSectionIndex: number;
132
- }) => Promise<void>;
133
- handleUpdateSolutions: (value: Record<string, SolutionDetails>) => void;
134
- handleUpdateCurrentAnswer: (value: SolutionDetails) => void;
135
- handleUpdateUserSettings: (value: UserSettings) => void;
136
- handleResetLocal: () => Promise<void>;
137
114
  test: Test | undefined;
138
- solutions: Record<string, SolutionDetails>;
115
+ solutions: Record<string, SolutionDetails$1>;
139
116
  currentQuestionIndex: number;
140
117
  currentSectionIndex: number;
141
- currentQuestionAnswer: SolutionDetails | undefined;
118
+ currentQuestionAnswer: SolutionDetails$1 | undefined;
142
119
  userSettings: UserSettings | undefined;
143
120
  isSolution?: boolean;
144
121
  submitting?: boolean;
122
+ timer: Record<string, number>;
123
+ isHydrated: boolean;
124
+ setTest: (test: Test, shouldUpdateSolution: boolean) => void;
125
+ setQuestionCoordinates: (sectionIndex: number, questionIndex: number) => void;
126
+ setSolution: (questionId: string, details: SolutionDetails$1) => void;
127
+ setCurrentAnswer: (answer: SolutionDetails$1 | undefined) => void;
128
+ setUserSettings: (settings: UserSettings) => void;
129
+ setTestSubmitting: (submitting: boolean) => void;
130
+ incrementTimer: (quesId: string) => void;
131
+ decrementTimer: (testId: string) => void;
132
+ reset: () => void;
133
+ setIsHydrated: (isHydrated: boolean) => void;
134
+ handleSubmitTest: (pause?: boolean, autoSubmit?: boolean) => Record<string, any>;
135
+ currentSection: Section | undefined;
136
+ currentQuestion: Question | undefined;
145
137
  };
146
138
 
147
- declare const useTestEngineBody: () => {
148
- isQuizAnswer: boolean;
149
- handleClearResponse: () => void;
150
- handleSelectOption: (optionId: string) => void;
151
- handleIntegerInputChange: (val: string) => void;
152
- handleMarkForReview: () => void;
153
- };
154
-
155
- declare const useTestEngineFooter: () => {
156
- handleNext: () => void;
157
- handlePrevious: () => void;
158
- handleSaveAnswer: () => void;
159
- handleMarkForReview: () => void;
160
- isLastQuestion: boolean;
161
- isFirstQuestion: boolean;
139
+ declare const useTestTimer: () => {
140
+ timer: number;
162
141
  };
163
142
 
164
- declare const useTestQuestionStatus: ({ includeAllSections, }?: {
143
+ type QuestionStatusType = "correct" | "incorrect" | "skipped";
144
+ declare const useQuestions: ({ includeAllSections, }?: {
165
145
  includeAllSections?: boolean;
166
146
  }) => {
167
147
  getQuestionStatus: (qi: number, si: number) => AttemptStatus.ATTEMPTED | AttemptStatus.NOT_ATTEMPTED | AttemptStatus.NOT_VISITED | AttemptStatus.MARKED_FOR_REVIEW | AttemptStatus.ATTEMPTED_AND_MARKED_FOR_REVIEW;
168
148
  questionStatusMap: Record<AttemptStatus, {
169
149
  count: number;
170
150
  }>;
171
- getAnsweredQuestionStatus: (i: number, si: number) => AttemptStatus.ATTEMPTED | AttemptStatus.NOT_ATTEMPTED | AttemptStatus.NOT_VISITED;
151
+ getAnsweredQuestionStatus: (qi: number, si: number) => AttemptStatus.ATTEMPTED | AttemptStatus.NOT_ATTEMPTED | AttemptStatus.NOT_VISITED;
172
152
  areArraysEqual: (a: string[], b: string[]) => boolean;
173
- };
174
-
175
- type QuestionStatusType = "correct" | "incorrect" | "skipped";
176
- declare const useTestSolutionBody: () => {
177
- correctAnswer: string;
178
- currentAnswerMarks: string | number;
179
153
  currentQuestionStatus: QuestionStatusType;
154
+ currentAnswerMarks: string | number;
155
+ correctAnswer: string;
180
156
  };
181
157
 
182
- declare const useTestTimer: (type: "web" | "app") => {
158
+ declare const useQuestionTimer: () => {
183
159
  timer: number;
184
160
  };
185
161
 
186
- declare const useQuestionStopWatch: (type?: "web" | "app") => {
187
- questionStopWatch: number;
162
+ declare const useTestActions: () => {
163
+ isQuizAnswer: boolean;
164
+ handleClearResponse: () => void;
165
+ handleSelectOption: (optionId: string) => void;
166
+ handleIntegerInputChange: (val: string) => void;
167
+ handleMarkForReview: () => void;
168
+ handleNext: () => void;
169
+ handlePrevious: () => void;
170
+ handleSaveAnswer: () => void;
171
+ isLastQuestion: boolean;
172
+ isFirstQuestion: boolean;
188
173
  };
189
174
 
190
- export { AttemptStatus, type SubmitTest, TestEngineProvider, useQuestionStopWatch, useTestEngine, useTestEngineBody, useTestEngineFooter, useTestQuestionStatus, useTestSolutionBody, useTestTimer };
175
+ interface SubmitTest {
176
+ payload: Record<string, any>;
177
+ pause: boolean;
178
+ autoSubmit: boolean;
179
+ }
180
+
181
+ interface ProviderRef {
182
+ reset: () => void;
183
+ }
184
+ declare const Provider: React.ForwardRefExoticComponent<{
185
+ test: Test;
186
+ isSolution?: boolean;
187
+ submitTest: React.Dispatch<SubmitTest>;
188
+ children: React.ReactNode;
189
+ } & React.RefAttributes<ProviderRef>>;
190
+
191
+ export { AttemptStatus, type ProviderRef, type SubmitTest, Provider as TestEngineProvider, useQuestionTimer, useQuestions, useTestActions, useTestEngine, useTestTimer };
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- "use strict";var ne=Object.create;var F=Object.defineProperty;var se=Object.getOwnPropertyDescriptor;var re=Object.getOwnPropertyNames;var ie=Object.getPrototypeOf,oe=Object.prototype.hasOwnProperty;var ue=(n,e)=>{for(var t in e)F(n,t,{get:e[t],enumerable:!0})},B=(n,e,t,o)=>{if(e&&typeof e=="object"||typeof e=="function")for(let s of re(e))!oe.call(n,s)&&s!==t&&F(n,s,{get:()=>e[s],enumerable:!(o=se(e,s))||o.enumerable});return n};var U=(n,e,t)=>(t=n!=null?ne(ie(n)):{},B(e||!n||!n.__esModule?F(t,"default",{value:n,enumerable:!0}):t,n)),ae=n=>B(F({},"__esModule",{value:!0}),n);var Ae={};ue(Ae,{AttemptStatus:()=>w,TestEngineProvider:()=>z,useQuestionStopWatch:()=>te,useTestEngine:()=>D,useTestEngineBody:()=>$,useTestEngineFooter:()=>j,useTestQuestionStatus:()=>Q,useTestSolutionBody:()=>X,useTestTimer:()=>ee});module.exports=ae(Ae);var Z=require("react");var K=U(require("react")),w=(r=>(r.ATTEMPTED="ATTEMPTED",r.NOT_ATTEMPTED="NOT_ATTEMPTED",r.NOT_VISITED="NOT_VISITED",r.MARKED_FOR_REVIEW="MARKED_FOR_REVIEW",r.ATTEMPTED_AND_MARKED_FOR_REVIEW="ATTEMPTED_AND_MARKED_FOR_REVIEW",r.ACITVE_QUESTION="ACITVE_QUESTION",r))(w||{});var W={test:void 0,solutions:{},currentQuestionIndex:0,currentSectionIndex:0,currentQuestionAnswer:{},userSettings:{},isSolution:!1,submitting:!1},P=K.default.createContext([W,()=>{},()=>{}]);P.displayName="TestEngineContext";var J=(n,e)=>{switch(e.type){case"UPDATE_TEST":return{...n,test:e.value};case"UPDATE_SOLUTIONS":return{...n,solutions:e.value};case"UPDATE_CURRENT_ANSWER":return{...n,currentQuestionAnswer:e.value};case"UPDATE_USER_SETTINGS":return{...n,userSettings:e.value};case"UPDATE_TEST_SUBMITTING":return{...n,submitting:e.value};case"UPDATE_QUESTION_COORDINATES":return{...n,currentSectionIndex:e.value.currentSectionIndex,currentQuestionIndex:e.value.currentQuestionIndex}}};var O=U(require("react"));var L=U(require("@react-native-async-storage/async-storage")),ce=()=>!(typeof navigator<"u"&&navigator.product==="ReactNative")&&typeof window<"u"&&window.localStorage?{getItem:e=>localStorage.getItem(e)?JSON.parse(localStorage.getItem(e)):null,setItem:(e,t)=>localStorage.setItem(e,JSON.stringify(t)),removeItem:e=>localStorage.removeItem(e)}:{getItem:async e=>{try{let t=await L.default.getItem(e);return t?JSON.parse(t):null}catch(t){return console.log(t),null}},setItem:async(e,t)=>{try{await L.default.setItem(e,JSON.stringify(t))}catch(o){console.log(o)}},removeItem:async e=>{try{await L.default.removeItem(e)}catch(t){console.log(t)}}},c=ce();var de=({test:n,submitTest:e,...t})=>{let[o,s]=O.default.useState(!1),[u,r]=O.default.useReducer(J,{...W,test:n,isSolution:t.isSolution}),T=async()=>{let l=await c.getItem(`${n?._id}-test-initiated`),m=await c.getItem(`${n._id}-solutions`)||{};r(l?{type:"UPDATE_SOLUTIONS",value:m}:{type:"UPDATE_SOLUTIONS",value:n?.sections?.reduce((i,E)=>(E.questions?.forEach(a=>{a.questionResponse&&(i[a._id]={...a.questionResponse})}),i),{})||{}});let d=await c.getItem(`${n._id}-currentSectionIndex`)||0,I=await c.getItem(`${n._id}-currentQuestionIndex`)||0;r({type:"UPDATE_QUESTION_COORDINATES",value:{currentSectionIndex:d,currentQuestionIndex:I}}),s(!0)};return(0,Z.useEffect)(()=>{T()},[]),o?O.default.createElement(P.Provider,{value:[u,r,e],...t}):O.default.createElement(O.default.Fragment,null)},z=de;var A=U(require("react"));var Te=()=>{let n=A.default.useContext(P);if(n===void 0)throw new Error("useTestEngine must be use within TestEngineProvider");let[e,t,o]=n,s=A.default.useMemo(()=>e.test?.sections[e.currentSectionIndex],[e.currentSectionIndex]),u=A.default.useMemo(()=>s?.questions[e.currentQuestionIndex],[e.currentQuestionIndex,s]);A.default.useEffect(()=>{u&&(e.solutions[u._id]?T({...e.solutions[u._id],isSaved:!0}):(e.isSolution||i({...e.solutions,[u._id]:{}}),T({}))),c.setItem(`${e.test?._id}-currentSectionIndex`,e.currentSectionIndex),c.setItem(`${e.test?._id}-currentQuestionIndex`,e.currentQuestionIndex)},[e.currentSectionIndex,e.currentQuestionIndex]);let r=A.default.useCallback(S=>t({type:"UPDATE_TEST",value:S}),[t]),T=A.default.useCallback(S=>t({type:"UPDATE_CURRENT_ANSWER",value:S}),[t]),l=A.default.useCallback(async S=>(await m(),t({type:"UPDATE_QUESTION_COORDINATES",value:S})),[t]),m=async()=>{let S=await c.getItem(`${e?.test?._id}-timers`)||{};e.test?.sections.forEach(v=>{v.questions?.forEach(R=>{S[R._id]&&!e.solutions[R._id]&&(e.solutions[R._id]={}),e.solutions[R._id]&&(e.solutions[R._id].timeTaken=S[R._id])})})},d=A.default.useCallback(S=>t({type:"UPDATE_USER_SETTINGS",value:S}),[t]),I=A.default.useCallback(S=>t({type:"UPDATE_TEST_SUBMITTING",value:S}),[t]);A.default.useEffect(()=>{e.isSolution||c.setItem(`${e.test?._id}-solutions`,e.solutions)},[e.solutions]);let i=A.default.useCallback(S=>t({type:"UPDATE_SOLUTIONS",value:S}),[t]),E=A.default.useCallback(async()=>{if(!await c.getItem(`${e.test?._id}-test-initiated`)&&e.test?.status==="Paused"){let v=await c.getItem(`${e?.test?._id}-timers`)||{},R={};e.test?.sections?.forEach((_,g)=>{_.questions?.forEach((p,y)=>{p._id===e.test?.lastVisitedQuestionId&&l({currentSectionIndex:g,currentQuestionIndex:y}),p.questionResponse&&(R[p._id]=JSON.stringify(p.questionResponse.timeTaken||0))})}),c.setItem(`${e?.test?._id}-timers`,{...v,...R})}e?.isSolution||c.setItem(`${e.test?._id}-test-initiated`,"true")},[]);A.default.useEffect(()=>{E()},[]);let a=async()=>{await c.removeItem(`${e.test?._id}-currentQuestionIndex`),await c.removeItem(`${e.test?._id}-currentSectionIndex`),await c.removeItem(`${e.test?._id}-solutions`),await c.removeItem(`${e.test?._id}-test-initiated`),await c.removeItem(`${e?.test?._id}-timers`)},f=A.default.useCallback(async(S=!1,v=!1)=>{I(!0);let R=await c.getItem(`${e.test?._id}-solutions`)||{},_=await c.getItem(`${e?.test?._id}-timers`)||{},g=[];e.test?.sections.forEach(y=>{y.questions.forEach(M=>{let C={questionId:M._id};R[M._id]&&(C.markedSolution=R[M._id].markedSolution||[],C.timeTaken=parseInt(_[M._id]||0),C.markedForReview=R[M._id].markedForReview||!1,g.push(C))})});let p={submittedBy:v?"autoSubmit":"user",questionResponses:g,testMappingId:e.test?.testMappingId,type:S?"Pause":"Submit",testId:e.test?._id};return S&&(p.lastVisitedQuestionId=u?._id),e.test?.timerLevel==="TEST"&&e.test.type==="TEST"&&S&&_[e?.test?._id]&&(p.timeTaken={[e.test?._id]:_[e?.test?._id]}),g.length||delete p.questionResponses,await a(),I(!1),o({payload:p,pause:S,autoSubmit:v}),p},[u]);return A.default.useMemo(()=>({...e,currentSection:s,currentQuestion:u,handleSubmitTest:f,handleUpdateTestDetails:r,handleUpdateQuestionCoordinates:l,handleUpdateSolutions:i,handleUpdateCurrentAnswer:T,handleUpdateUserSettings:d,handleResetLocal:a}),[e,s,u,f,r,l,i,T,d,a])},D=Te;var N=U(require("react"));var Ee=()=>{let{currentQuestion:n,currentQuestionAnswer:e,test:t,solutions:o,isSolution:s,handleUpdateCurrentAnswer:u,handleUpdateSolutions:r}=D(),T=N.default.useMemo(()=>t?.type==="QUIZ"&&!!o[n?._id]?.markedSolution?.length,[t,o,n]),l=(0,N.useCallback)(()=>{u({...e,markedSolution:[],isSaved:!1}),r({...o,[n?._id]:{...e,markedSolution:[]}})},[e,n,o,u,r]),m=(0,N.useCallback)(i=>{if(!(s||T))if(e?.markedSolution?.includes(i)){let E=e.markedSolution.filter(a=>a!==i);return E?.length===0?l():u({...e,markedSolution:E,isSaved:!1})}else{if(n?.type==="SINGLE")return u({...e,markedSolution:[i],isSaved:!1});{let E=e?.markedSolution?.length?e?.markedSolution:[];return u({...e,markedSolution:[...E,i],isSaved:!1})}}},[s,T,n,e,u,l]),d=(0,N.useCallback)(i=>{s||T||(u({...e,markedSolution:i?[i]:[],isSaved:!1}),i||l())},[s,T,u,e,l]),I=(0,N.useCallback)(()=>{u({...e,markedForReview:!e?.markedForReview}),r({...o,[n._id]:{...o[n._id],markedForReview:!e?.markedForReview}})},[u,r,e,o,n]);return{isQuizAnswer:T,handleClearResponse:l,handleSelectOption:m,handleIntegerInputChange:d,handleMarkForReview:I}},$=Ee;var k=require("react");var G=U(require("react"));var Ie=({includeAllSections:n}={})=>{let{currentQuestionIndex:e,solutions:t,currentSectionIndex:o,test:s,isSolution:u}=D(),[r,T]=G.default.useState({MARKED_FOR_REVIEW:{count:0},ATTEMPTED:{count:0},ATTEMPTED_AND_MARKED_FOR_REVIEW:{count:0},NOT_ATTEMPTED:{count:0},NOT_VISITED:{count:0},ACITVE_QUESTION:{count:0}}),l=()=>{let i={...r};Object.values(i).forEach(E=>E.count=0),n?s?.sections?.forEach((E,a)=>{E?.questions?.forEach((f,b)=>{let S=u||s.type==="QUIZ"?I(b,a):m(b,a);i[S].count+=1})}):s?.sections[o]?.questions.forEach((E,a)=>{let f=u||s.type==="QUIZ"?I(a,o):m(a,o);i[f].count+=1}),T(i)};G.default.useEffect(()=>{l()},[e,t,o]);let m=(i,E)=>{let a=s?.sections[E]?.questions[i],f;return t[a?._id]&&t[a?._id].markedSolution?.length&&t[a?._id].markedForReview?f="ATTEMPTED_AND_MARKED_FOR_REVIEW":t[a?._id]&&t[a?._id].markedForReview?f="MARKED_FOR_REVIEW":t[a?._id]&&t[a?._id].markedSolution?.length?f="ATTEMPTED":t[a?._id]?f="NOT_ATTEMPTED":f="NOT_VISITED",f};function d(i,E){let a=i.slice().sort(),f=E.slice().sort();if(a.length!==f.length)return!1;for(let b=0;b<a.length;b++)if(a[b]!==f[b])return!1;return!0}let I=(i,E)=>{let a=s?.sections[E]?.questions[i],f;return t[a?._id]&&t[a?._id].markedSolution?.length?d(t[a?._id].markedSolution,a.solution)?f="ATTEMPTED":f="NOT_ATTEMPTED":f="NOT_VISITED",f};return{getQuestionStatus:m,questionStatusMap:r,getAnsweredQuestionStatus:I,areArraysEqual:d}},Q=Ie;var fe=()=>{let{handleUpdateQuestionCoordinates:n,currentQuestionIndex:e,currentSectionIndex:t,currentSection:o,currentQuestionAnswer:s,handleUpdateSolutions:u,solutions:r,currentQuestion:T,handleUpdateCurrentAnswer:l,test:m,userSettings:d}=D(),{getAnsweredQuestionStatus:I}=Q(),[i,E]=(0,k.useState)([]);(0,k.useEffect)(()=>{if(d?.selectedFilter){let _=o?.questions?.map((g,p)=>{if(d?.selectedFilter===I(p,t))return p})?.filter(g=>g!==void 0);E(_),_?.length&&n({currentSectionIndex:t,currentQuestionIndex:_[0]||0})}else E([])},[d?.selectedFilter,o]);let a=()=>{s?.isSaved||(delete s?.isSaved,u({...r,[T?._id]:{...s}}))},f=()=>{if(d?.selectedFilter){let p=i?.findIndex(y=>y===e);p>=0&&i[p+1]>=0&&n({currentSectionIndex:t,currentQuestionIndex:i[p+1]});return}let _=o?.questions[e+1]?t:m?.sections[t+1]?t+1:t,g=o?.questions[e+1]?e+1:0;n({currentSectionIndex:_,currentQuestionIndex:g})},b=()=>{if(d?.selectedFilter){let p=i?.findIndex(y=>y===e);p>=0&&i[p-1]>=0&&n({currentSectionIndex:t,currentQuestionIndex:i[p-1]});return}let _=o?.questions[e-1]?t:m?.sections[t-1]?t-1:t,g=o?.questions[e-1]?e-1:m.sections[_]?.questions?.length-1;n({currentSectionIndex:_,currentQuestionIndex:g})},S=(0,k.useMemo)(()=>{if(d?.selectedFilter){let _=i?.findIndex(g=>g===e);return!i[_+1]}return!m?.sections[t+1]&&!o?.questions[e+1]},[t,e,i,d?.selectedFilter]),v=(0,k.useMemo)(()=>d?.selectedFilter?i?.findIndex(g=>g===e)<=0:t===0&&e===0,[t,e,i,d?.selectedFilter]);return{handleNext:f,handlePrevious:b,handleSaveAnswer:a,handleMarkForReview:()=>{l({...s,markedForReview:!s?.markedForReview}),u({...r,[T._id]:{...r[T._id],markedForReview:!s?.markedForReview}})},isLastQuestion:S,isFirstQuestion:v}},j=fe;var V=U(require("react"));var H={1:{label:"A"},2:{label:"B"},3:{label:"C"},4:{label:"D"}},pe=()=>{let{currentQuestionIndex:n,test:e,currentSection:t,currentSectionIndex:o,currentQuestion:s}=D(),{getAnsweredQuestionStatus:u}=Q(),r=V.default.useMemo(()=>u(n,o)==="ATTEMPTED"?"correct":u(n,o)==="NOT_ATTEMPTED"?"incorrect":"skipped",[n,o]),T=V.default.useMemo(()=>{switch(r){case"correct":return e?.marksLevel==="TEST"?"+"+e.positiveMarks:"+"+t?.positiveMarks;case"incorrect":return e?.marksLevel==="TEST"?e.isNegativeMarkingEnabled?"-"+e.negativeMarks:0:t?.isNegativeMarkingEnabled?"-"+t?.negativeMarks:0;case"skipped":return 0}},[r,o,n]);return{correctAnswer:V.default.useMemo(()=>{if(!s)return"";if(s.type==="INTEGER")return s.solution[0];if(s.type==="SINGLE"){let m=s.options.findIndex(d=>d._id===s.solution[0]);return H[m+1]?.label||""}return s.type==="MULTIPLE"?s.solution.map(d=>{let I=s.options.findIndex(i=>i._id===d);return H[I+1]?.label||""}).join(", "):""},[s]),currentAnswerMarks:T,currentQuestionStatus:r}},X=pe;var q=U(require("react"));var _e=n=>{let{test:e,handleSubmitTest:t}=D(),o=q.default.useRef(null),[s,u]=q.default.useState(e?.timeTaken?.[e?._id]??e?.duration??0),r=async l=>{let m=await c.getItem(`${e?._id}-timers`);if(l<=0){t(!1,!0);return}m[e._id]=l+"",c.setItem(`${e?._id}-timers`,m),u(l),o.current=setTimeout(()=>{r(l-1)},n==="app"?950:1e3)},T=async()=>{let l=await c.getItem(`${e?._id}-timers`);l||(await c.setItem(`${e?._id}-timers`,{}),l={});let m=parseInt(l?.[e?._id]||e?.timeTaken?.[e?._id]||e?.duration||0);r(m)};return q.default.useEffect(()=>(T(),()=>{o.current&&clearTimeout(o.current)}),[]),{timer:s}},ee=_e;var h=U(require("react"));var ge=n=>{let e=n==="app"?950:1e3,{currentQuestion:t,solutions:o,test:s}=D(),{isQuizAnswer:u}=$(),r=h.default.useRef(null),[T,l]=h.default.useState(0),m=async I=>{if(!t)return;let i=await c.getItem(`${s?._id}-timers`);if((o[t?._id]?.markedSolution||[])?.length===0)i[t?._id]=I+"",c.setItem(`${s?._id}-timers`,i),l(I);else{clearTimeout(r.current);return}r.current=setTimeout(()=>{m(I+1)},e)},d=async()=>{r.current&&(clearTimeout(r.current),r.current=null);let I=await c.getItem(`${s?._id}-timers`);I||(await c.setItem(`${s?._id}-timers`,{}),I={});let i=parseInt(I?.[t?._id]||s?.timeTaken?.[t?._id]||0);l(i),m(i)};return h.default.useEffect(()=>(t?._id&&d(),()=>{r.current&&(clearTimeout(r.current),r.current=null)}),[t]),h.default.useEffect(()=>{r.current===null&&setTimeout(()=>{r.current===null&&d()},e)},[o]),h.default.useEffect(()=>{u&&(clearTimeout(r.current),r.current=null)},[u]),{questionStopWatch:T}},te=ge;0&&(module.exports={AttemptStatus,TestEngineProvider,useQuestionStopWatch,useTestEngine,useTestEngineBody,useTestEngineFooter,useTestQuestionStatus,useTestSolutionBody,useTestTimer});
1
+ "use strict";var re=Object.create;var P=Object.defineProperty;var se=Object.getOwnPropertyDescriptor;var oe=Object.getOwnPropertyNames;var ie=Object.getPrototypeOf,ue=Object.prototype.hasOwnProperty;var ae=(t,e)=>{for(var n in e)P(t,n,{get:e[n],enumerable:!0})},z=(t,e,n,s)=>{if(e&&typeof e=="object"||typeof e=="function")for(let o of oe(e))!ue.call(t,o)&&o!==n&&P(t,o,{get:()=>e[o],enumerable:!(s=se(e,o))||s.enumerable});return t};var O=(t,e,n)=>(n=t!=null?re(ie(t)):{},z(e||!t||!t.__esModule?P(n,"default",{value:t,enumerable:!0}):n,t)),ce=t=>z(P({},"__esModule",{value:!0}),t);var _e={};ae(_e,{AttemptStatus:()=>V,TestEngineProvider:()=>ee,useQuestionTimer:()=>J,useQuestions:()=>U,useTestActions:()=>X,useTestEngine:()=>k,useTestTimer:()=>Y});module.exports=ce(_e);var F=O(require("react"));var B=require("zustand"),C=require("zustand/middleware");var de=()=>typeof globalThis>"u"||!globalThis.localStorage?null:{setItem:(t,e)=>{globalThis.localStorage.setItem(t,e)},getItem:t=>globalThis.localStorage.getItem(t)??null,removeItem:t=>{globalThis.localStorage.removeItem(t)}},le=()=>{try{let{createMMKV:t}=require("react-native-mmkv"),e=t();return{setItem:(n,s)=>{e.set(n,s)},getItem:n=>e.getString(n)??null,removeItem:n=>{e.delete(n)}}}catch{return null}},K={},me=()=>({setItem:(t,e)=>{K[t]=e},getItem:t=>K[t]??null,removeItem:t=>{delete K[t]}}),h=de()||le()||me();var Z=O(require("react")),W={test:void 0,solutions:{},currentQuestionIndex:0,currentSectionIndex:0,currentQuestionAnswer:{},userSettings:{},isSolution:!1,submitting:!1,timer:{},isHydrated:!1},w=Z.default.createContext([W,()=>{}]);w.displayName="TestEngineContext";var i=(0,B.create)()((0,C.persist)(t=>({...W,setQuestionCoordinates:(e,n)=>t(s=>{let r=s.test?.sections[e]?.questions[n]?._id,c=null;return s.isSolution||(s.solutions[r]?c={...s.solutions,[r]:{...s.solutions[r],timeTaken:s.timer[r]||0}}:c={...s.solutions,[r]:{}}),{currentSectionIndex:e,currentQuestionIndex:n,currentQuestionAnswer:c?.[r]||{},...c&&{solutions:c}}}),setTest:(e,n)=>t(()=>{if(!n)return{test:e};let s={},o={},u=0,r=0;e?.sections?.forEach((E,p)=>{E?.questions?.forEach((T,A)=>{T?.questionResponse&&(s={...s,[T?._id]:{markedSolution:T?.questionResponse?.markedSolution||[],markedForReview:T?.questionResponse?.markedForReview||!1,timeTaken:T?.questionResponse?.timeTaken||0,isSaved:!0}},o={...o,[T?._id]:T?.questionResponse?.timeTaken||0}),e?.status==="Paused"&&T._id===e?.lastVisitedQuestionId&&(u=p,r=A)})}),o[e?._id]=e?.timeTaken&&+e?.timeTaken[e?._id]||+e?.duration||0;let c=e?.sections[u]?.questions[r]?._id,m=s[c]||{};return{test:e,solutions:s,timer:o,currentSectionIndex:u,currentQuestionIndex:r,currentQuestionAnswer:m}}),setSolution:(e,n)=>t(s=>({solutions:{...s.solutions,[e]:{...s.solutions[e],...n}},currentQuestionAnswer:{...s.solutions[e],...n}})),incrementTimer:e=>t(n=>({timer:{...n.timer,[e]:(n.timer[e]||0)+1}})),decrementTimer:e=>t(n=>({timer:{...n.timer,[e]:(n.timer[e]||0)-1}})),setCurrentAnswer:e=>t({currentQuestionAnswer:e}),setUserSettings:e=>t({userSettings:e}),setTestSubmitting:e=>t({submitting:e}),reset:()=>t(W),setIsHydrated:e=>t({isHydrated:e})}),{name:"test-engine-v2-store",storage:(0,C.createJSONStorage)(()=>h)}));var Te=()=>{let t=F.default.useContext(w);if(t===void 0)throw new Error("useTestEngineV2 must be use within TestEngineProvider");let[e,n]=t;if(!e.test)throw new Error("Please provide test data to useTestEngineV2");let s=i(S=>S.test),o=i(S=>S.setTestSubmitting),u=i(S=>S.solutions),r=i(S=>S.currentQuestionIndex),c=i(S=>S.currentSectionIndex),m=(0,F.useMemo)(()=>s?.sections[c],[s,c]),E=(0,F.useMemo)(()=>m?.questions[r],[m,r]),p=i(S=>S.reset),T=i(S=>S.timer);return{handleSubmitTest:(S=!1,_=!1)=>{o(!0);let x=[];s?.sections?.forEach(f=>{f?.questions?.forEach(d=>{u[d._id]&&x.push({questionId:d._id,markedSolution:u[d._id]?.isSaved?u[d._id]?.markedSolution:[],timeTaken:T[d._id]||u[d._id]?.timeTaken||0,markedForReview:u[d._id]?.markedForReview||!1})})});let Q={submittedBy:_?"autoSubmit":"user",questionResponses:x,testMappingId:s?.testMappingId,type:S?"Pause":"Submit",testId:s?._id,...S?{lastVisitedQuestionId:E?._id}:{},...s?.type==="TEST"&&S&&T[s?._id]?{timeTaken:{[s?._id]:T[s?._id]}}:{}};return x.length||delete Q.questionResponses,o(!1),n({payload:Q,pause:S,autoSubmit:_}),Q},currentSection:m,currentQuestion:E,...i.getState(),...e}},k=Te;var N=require("react");var Se=()=>{let{isSolution:t,handleSubmitTest:e}=k(),n=i(m=>m.test),s=i(m=>m.timer),o=i(m=>m.decrementTimer),u=i(m=>m.isHydrated),r=(0,N.useRef)(null),c=()=>{n&&u&&o(n?._id),r.current=setTimeout(()=>{c()},1e3)};return(0,N.useEffect)(()=>{if(n&&s[n?._id]<=0){clearTimeout(r.current),e(!1,!0);return}},[s,n]),(0,N.useEffect)(()=>{if(!t)return c(),()=>{r.current&&clearTimeout(r.current)}},[n,u]),{timer:s[n._id]||0}},Y=Se;var M=O(require("react"));var V=(r=>(r.ATTEMPTED="ATTEMPTED",r.NOT_ATTEMPTED="NOT_ATTEMPTED",r.NOT_VISITED="NOT_VISITED",r.MARKED_FOR_REVIEW="MARKED_FOR_REVIEW",r.ATTEMPTED_AND_MARKED_FOR_REVIEW="ATTEMPTED_AND_MARKED_FOR_REVIEW",r.ACITVE_QUESTION="ACITVE_QUESTION",r))(V||{});var j={1:{label:"A"},2:{label:"B"},3:{label:"C"},4:{label:"D"}},ge=({includeAllSections:t}={})=>{let e=i(f=>f.test),n=i(f=>f.solutions),s=i(f=>f.isSolution),o=i(f=>f.currentSectionIndex),{currentQuestion:u,currentSection:r}=k(),c=i(f=>f.currentQuestionIndex),[m,E]=M.default.useState({MARKED_FOR_REVIEW:{count:0},ATTEMPTED:{count:0},ATTEMPTED_AND_MARKED_FOR_REVIEW:{count:0},NOT_ATTEMPTED:{count:0},NOT_VISITED:{count:0},ACITVE_QUESTION:{count:0}}),p=(0,M.useCallback)((f,d)=>{let a=e?.sections[d]?.questions[f],g=n[a?._id];return g?g?.markedSolution&&g?.markedSolution?.length>0&&g?.markedForReview?"ATTEMPTED_AND_MARKED_FOR_REVIEW":g?.markedSolution&&g?.markedSolution?.length>0?"ATTEMPTED":g?.markedForReview?"MARKED_FOR_REVIEW":"NOT_ATTEMPTED":"NOT_VISITED"},[n,e]),T=(0,M.useCallback)((f,d)=>{let a=e?.sections[d]?.questions[f],g=n[a?._id];return g?.markedSolution?.length?S(g.markedSolution,a.solution)?"ATTEMPTED":"NOT_ATTEMPTED":"NOT_VISITED"},[n,e]),A=(0,M.useCallback)(()=>{let f={...m};Object.values(f).forEach(d=>d.count=0),t?e?.sections?.forEach((d,a)=>{d?.questions?.forEach((g,D)=>{let H=s||e.type==="QUIZ"?T(D,a):p(D,a);f[H].count+=1})}):e?.sections[o]?.questions.forEach((d,a)=>{let g=s||e.type==="QUIZ"?T(a,o):p(a,o);f[g].count+=1}),E(f)},[T,p,t,s,e,o,n]);M.default.useEffect(()=>{A()},[c,n,o]);function S(f,d){let a=f.slice().sort(),g=d.slice().sort();if(a.length!==g.length)return!1;for(let D=0;D<a.length;D++)if(a[D]!==g[D])return!1;return!0}let _=M.default.useMemo(()=>T(c,o)==="ATTEMPTED"?"correct":T(c,o)==="NOT_ATTEMPTED"?"incorrect":"skipped",[c,o,T]),x=M.default.useMemo(()=>{switch(_){case"correct":return e?.marksLevel==="TEST"?"+"+e.positiveMarks:"+"+r?.positiveMarks;case"incorrect":return e?.marksLevel==="TEST"?e.isNegativeMarkingEnabled?"-"+e.negativeMarks:0:r?.isNegativeMarkingEnabled?"-"+r?.negativeMarks:0;case"skipped":return 0}},[_]),Q=M.default.useMemo(()=>{if(!u)return"";if(u.type==="INTEGER")return u.solution[0];if(u.type==="SINGLE"){let f=u.options.findIndex(d=>d._id===u.solution[0]);return j[f+1]?.label||""}return u.type==="MULTIPLE"?u.solution.map(d=>{let a=u.options.findIndex(g=>g._id===d);return j[a+1]?.label||""}).join(", "):""},[u]);return{getQuestionStatus:p,questionStatusMap:m,getAnsweredQuestionStatus:T,areArraysEqual:S,currentQuestionStatus:_,currentAnswerMarks:x,correctAnswer:Q}},U=ge;var q=require("react");var Ee=()=>{let{currentQuestion:t,isSolution:e}=k(),n=i(m=>m.timer),s=i(m=>m.incrementTimer),o=i(m=>m.solutions),u=i(m=>m.isHydrated),r=(0,q.useRef)(null),c=()=>{t&&!o[t?._id]?.markedSolution?.length&&u&&s(t?._id),r.current=setTimeout(()=>{c()},1e3)};return(0,q.useEffect)(()=>{if(!e)return c(),()=>{r.current&&clearTimeout(r.current)}},[t,o,u]),{timer:n[t._id]||0}},J=Ee;var I=O(require("react"));var pe=()=>{let t=i(l=>l.test),e=i(l=>l.solutions),{currentQuestion:n,currentSection:s,currentQuestionAnswer:o}=k(),u=i(l=>l.currentQuestionIndex),r=i(l=>l.currentSectionIndex),c=i(l=>l.userSettings),m=i(l=>l.isSolution),E=i(l=>l.setQuestionCoordinates),p=i(l=>l.setSolution),T=i(l=>l.setCurrentAnswer),{getAnsweredQuestionStatus:A}=U(),S=I.default.useMemo(()=>t?.type==="QUIZ"&&!!e[n?._id]?.markedSolution?.length,[t,e,n]),_=(0,I.useCallback)(()=>{T({...o,markedSolution:[],isSaved:!1}),p(n?._id,{...o,markedSolution:[]})},[o,n,p]),x=(0,I.useCallback)(l=>{if(!(m||S))if(o?.markedSolution?.includes(l)){let b=o.markedSolution.filter(R=>R!==l);if(b?.length===0)return _();T({...o,markedSolution:b,isSaved:!1});return}else if(n?.type==="SINGLE"){T({...o,markedSolution:[l],isSaved:!1});return}else{let b=o?.markedSolution?.length?o?.markedSolution:[];T({...o,markedSolution:[...b,l],isSaved:!1});return}},[m,S,n,o,T,_]),Q=(0,I.useCallback)(l=>{m||S||(T({...o,markedSolution:l?[l]:[],isSaved:!1}),l||_())},[m,S,o,_]),f=(0,I.useCallback)(()=>{T({...o,markedForReview:!o?.markedForReview}),p(n._id,{...e[n._id],markedForReview:!o?.markedForReview})},[p,o,e,n]),[d,a]=(0,I.useState)([]);(0,I.useEffect)(()=>{if(c?.selectedFilter){let l=s?.questions?.map((b,R)=>{if(c?.selectedFilter===A(R,r))return R})?.filter(b=>b!==void 0);a(l),l?.length&&E(r,l[0]||0)}else a([])},[c,s,A,E]);let g=(0,I.useCallback)(()=>{o?.isSaved||p(n?._id,{...o,isSaved:!0})},[o,n,p]),D=(0,I.useCallback)(()=>{if(c?.selectedFilter){let R=d?.findIndex(L=>L===u);R>=0&&d[R+1]>=0&&E(r,d[R+1]);return}let l=s?.questions[u+1]?r:t?.sections[r+1]?r+1:r,b=s?.questions[u+1]?u+1:0;E(l,b)},[c?.selectedFilter,d,u,r,s,t,E]),H=(0,I.useCallback)(()=>{if(c?.selectedFilter){let R=d?.findIndex(L=>L===u);R>=0&&d[R-1]>=0&&E(r,d[R-1]);return}let l=s?.questions[u-1]?r:t?.sections[r-1]?r-1:r,b=s?.questions[u-1]?u-1:t.sections[l]?.questions?.length-1;E(l,b)},[c?.selectedFilter,d,u,r,s,t,E]),te=(0,I.useMemo)(()=>{if(c?.selectedFilter){let l=d?.findIndex(b=>b===u);return!d[l+1]}return!t?.sections[r+1]&&!s?.questions[u+1]},[r,u,d,c?.selectedFilter,t,s]),ne=(0,I.useMemo)(()=>c?.selectedFilter?d?.findIndex(b=>b===u)<=0:r===0&&u===0,[r,u,d,c?.selectedFilter]);return{isQuizAnswer:S,handleClearResponse:_,handleSelectOption:x,handleIntegerInputChange:Q,handleMarkForReview:f,handleNext:D,handlePrevious:H,handleSaveAnswer:g,isLastQuestion:te,isFirstQuestion:ne}},X=pe;var y=require("react"),G=O(require("react"));var Ie=(0,y.forwardRef)(({test:t,isSolution:e,submitTest:n,children:s},o)=>{let u=i(a=>a.setTest),r=i(a=>a.setIsHydrated),c=i(a=>a.isHydrated),m=i(a=>a.test),E=i(a=>a.solutions),p=i(a=>a.currentQuestionIndex),T=i(a=>a.currentSectionIndex),A=i(a=>a.currentQuestionAnswer),S=i(a=>a.userSettings),_=i(a=>a.submitting),x=i(a=>a.timer),Q=i(a=>a.reset),f=G.default.useMemo(()=>({test:m,solutions:E,currentQuestionIndex:p,currentSectionIndex:T,userSettings:S,isSolution:e||!1,submitting:_,timer:x,isHydrated:c,currentQuestionAnswer:A}),[m,E,p,T,S,e,_,x,c,A]),d=()=>{console.log("[RESET]: Called"),Q(),h.removeItem(`${t?._id}-test-initiated`),h.removeItem(`test-engine-v2-store-${t?._id}`),h.removeItem("test-engine-v2-store")};return(0,y.useImperativeHandle)(o,()=>({reset:d})),(0,y.useEffect)(()=>{(async()=>{if(!(c&&t?._id===m?._id))try{h.getItem(`${t?._id}-test-initiated`)?u(t,!1):(u(t,!0),e||h.setItem(`${t?._id}-test-initiated`,"true")),i.persist.setOptions({name:`test-engine-v2-store-${t?._id}`}),await i.persist.rehydrate(),h.removeItem("test-engine-v2-store"),r(!0)}catch(g){console.error("Error initializing test engine store:",g),r(!0)}})()},[t?._id,c,u,r]),!c||!m?null:G.default.createElement(w.Provider,{value:[f,n]},s)}),ee=Ie;0&&(module.exports={AttemptStatus,TestEngineProvider,useQuestionTimer,useQuestions,useTestActions,useTestEngine,useTestTimer});
package/dist/index.mjs CHANGED
@@ -1 +1 @@
1
- import{useEffect as Y}from"react";import J from"react";var w=(n=>(n.ATTEMPTED="ATTEMPTED",n.NOT_ATTEMPTED="NOT_ATTEMPTED",n.NOT_VISITED="NOT_VISITED",n.MARKED_FOR_REVIEW="MARKED_FOR_REVIEW",n.ATTEMPTED_AND_MARKED_FOR_REVIEW="ATTEMPTED_AND_MARKED_FOR_REVIEW",n.ACITVE_QUESTION="ACITVE_QUESTION",n))(w||{});var C={test:void 0,solutions:{},currentQuestionIndex:0,currentSectionIndex:0,currentQuestionAnswer:{},userSettings:{},isSolution:!1,submitting:!1},k=J.createContext([C,()=>{},()=>{}]);k.displayName="TestEngineContext";var q=(r,e)=>{switch(e.type){case"UPDATE_TEST":return{...r,test:e.value};case"UPDATE_SOLUTIONS":return{...r,solutions:e.value};case"UPDATE_CURRENT_ANSWER":return{...r,currentQuestionAnswer:e.value};case"UPDATE_USER_SETTINGS":return{...r,userSettings:e.value};case"UPDATE_TEST_SUBMITTING":return{...r,submitting:e.value};case"UPDATE_QUESTION_COORDINATES":return{...r,currentSectionIndex:e.value.currentSectionIndex,currentQuestionIndex:e.value.currentQuestionIndex}}};import O from"react";import F from"@react-native-async-storage/async-storage";var Z=()=>!(typeof navigator<"u"&&navigator.product==="ReactNative")&&typeof window<"u"&&window.localStorage?{getItem:e=>localStorage.getItem(e)?JSON.parse(localStorage.getItem(e)):null,setItem:(e,t)=>localStorage.setItem(e,JSON.stringify(t)),removeItem:e=>localStorage.removeItem(e)}:{getItem:async e=>{try{let t=await F.getItem(e);return t?JSON.parse(t):null}catch(t){return console.log(t),null}},setItem:async(e,t)=>{try{await F.setItem(e,JSON.stringify(t))}catch(o){console.log(o)}},removeItem:async e=>{try{await F.removeItem(e)}catch(t){console.log(t)}}},c=Z();var j=({test:r,submitTest:e,...t})=>{let[o,i]=O.useState(!1),[u,n]=O.useReducer(q,{...C,test:r,isSolution:t.isSolution}),T=async()=>{let l=await c.getItem(`${r?._id}-test-initiated`),m=await c.getItem(`${r._id}-solutions`)||{};n(l?{type:"UPDATE_SOLUTIONS",value:m}:{type:"UPDATE_SOLUTIONS",value:r?.sections?.reduce((s,E)=>(E.questions?.forEach(a=>{a.questionResponse&&(s[a._id]={...a.questionResponse})}),s),{})||{}});let d=await c.getItem(`${r._id}-currentSectionIndex`)||0,I=await c.getItem(`${r._id}-currentQuestionIndex`)||0;n({type:"UPDATE_QUESTION_COORDINATES",value:{currentSectionIndex:d,currentQuestionIndex:I}}),i(!0)};return Y(()=>{T()},[]),o?O.createElement(k.Provider,{value:[u,n,e],...t}):O.createElement(O.Fragment,null)},H=j;import A from"react";var ee=()=>{let r=A.useContext(k);if(r===void 0)throw new Error("useTestEngine must be use within TestEngineProvider");let[e,t,o]=r,i=A.useMemo(()=>e.test?.sections[e.currentSectionIndex],[e.currentSectionIndex]),u=A.useMemo(()=>i?.questions[e.currentQuestionIndex],[e.currentQuestionIndex,i]);A.useEffect(()=>{u&&(e.solutions[u._id]?T({...e.solutions[u._id],isSaved:!0}):(e.isSolution||s({...e.solutions,[u._id]:{}}),T({}))),c.setItem(`${e.test?._id}-currentSectionIndex`,e.currentSectionIndex),c.setItem(`${e.test?._id}-currentQuestionIndex`,e.currentQuestionIndex)},[e.currentSectionIndex,e.currentQuestionIndex]);let n=A.useCallback(S=>t({type:"UPDATE_TEST",value:S}),[t]),T=A.useCallback(S=>t({type:"UPDATE_CURRENT_ANSWER",value:S}),[t]),l=A.useCallback(async S=>(await m(),t({type:"UPDATE_QUESTION_COORDINATES",value:S})),[t]),m=async()=>{let S=await c.getItem(`${e?.test?._id}-timers`)||{};e.test?.sections.forEach(v=>{v.questions?.forEach(R=>{S[R._id]&&!e.solutions[R._id]&&(e.solutions[R._id]={}),e.solutions[R._id]&&(e.solutions[R._id].timeTaken=S[R._id])})})},d=A.useCallback(S=>t({type:"UPDATE_USER_SETTINGS",value:S}),[t]),I=A.useCallback(S=>t({type:"UPDATE_TEST_SUBMITTING",value:S}),[t]);A.useEffect(()=>{e.isSolution||c.setItem(`${e.test?._id}-solutions`,e.solutions)},[e.solutions]);let s=A.useCallback(S=>t({type:"UPDATE_SOLUTIONS",value:S}),[t]),E=A.useCallback(async()=>{if(!await c.getItem(`${e.test?._id}-test-initiated`)&&e.test?.status==="Paused"){let v=await c.getItem(`${e?.test?._id}-timers`)||{},R={};e.test?.sections?.forEach((_,g)=>{_.questions?.forEach((p,y)=>{p._id===e.test?.lastVisitedQuestionId&&l({currentSectionIndex:g,currentQuestionIndex:y}),p.questionResponse&&(R[p._id]=JSON.stringify(p.questionResponse.timeTaken||0))})}),c.setItem(`${e?.test?._id}-timers`,{...v,...R})}e?.isSolution||c.setItem(`${e.test?._id}-test-initiated`,"true")},[]);A.useEffect(()=>{E()},[]);let a=async()=>{await c.removeItem(`${e.test?._id}-currentQuestionIndex`),await c.removeItem(`${e.test?._id}-currentSectionIndex`),await c.removeItem(`${e.test?._id}-solutions`),await c.removeItem(`${e.test?._id}-test-initiated`),await c.removeItem(`${e?.test?._id}-timers`)},f=A.useCallback(async(S=!1,v=!1)=>{I(!0);let R=await c.getItem(`${e.test?._id}-solutions`)||{},_=await c.getItem(`${e?.test?._id}-timers`)||{},g=[];e.test?.sections.forEach(y=>{y.questions.forEach(N=>{let M={questionId:N._id};R[N._id]&&(M.markedSolution=R[N._id].markedSolution||[],M.timeTaken=parseInt(_[N._id]||0),M.markedForReview=R[N._id].markedForReview||!1,g.push(M))})});let p={submittedBy:v?"autoSubmit":"user",questionResponses:g,testMappingId:e.test?.testMappingId,type:S?"Pause":"Submit",testId:e.test?._id};return S&&(p.lastVisitedQuestionId=u?._id),e.test?.timerLevel==="TEST"&&e.test.type==="TEST"&&S&&_[e?.test?._id]&&(p.timeTaken={[e.test?._id]:_[e?.test?._id]}),g.length||delete p.questionResponses,await a(),I(!1),o({payload:p,pause:S,autoSubmit:v}),p},[u]);return A.useMemo(()=>({...e,currentSection:i,currentQuestion:u,handleSubmitTest:f,handleUpdateTestDetails:n,handleUpdateQuestionCoordinates:l,handleUpdateSolutions:s,handleUpdateCurrentAnswer:T,handleUpdateUserSettings:d,handleResetLocal:a}),[e,i,u,f,n,l,s,T,d,a])},D=ee;import ne,{useCallback as P}from"react";var se=()=>{let{currentQuestion:r,currentQuestionAnswer:e,test:t,solutions:o,isSolution:i,handleUpdateCurrentAnswer:u,handleUpdateSolutions:n}=D(),T=ne.useMemo(()=>t?.type==="QUIZ"&&!!o[r?._id]?.markedSolution?.length,[t,o,r]),l=P(()=>{u({...e,markedSolution:[],isSaved:!1}),n({...o,[r?._id]:{...e,markedSolution:[]}})},[e,r,o,u,n]),m=P(s=>{if(!(i||T))if(e?.markedSolution?.includes(s)){let E=e.markedSolution.filter(a=>a!==s);return E?.length===0?l():u({...e,markedSolution:E,isSaved:!1})}else{if(r?.type==="SINGLE")return u({...e,markedSolution:[s],isSaved:!1});{let E=e?.markedSolution?.length?e?.markedSolution:[];return u({...e,markedSolution:[...E,s],isSaved:!1})}}},[i,T,r,e,u,l]),d=P(s=>{i||T||(u({...e,markedSolution:s?[s]:[],isSaved:!1}),s||l())},[i,T,u,e,l]),I=P(()=>{u({...e,markedForReview:!e?.markedForReview}),n({...o,[r._id]:{...o[r._id],markedForReview:!e?.markedForReview}})},[u,n,e,o,r]);return{isQuizAnswer:T,handleClearResponse:l,handleSelectOption:m,handleIntegerInputChange:d,handleMarkForReview:I}},L=se;import{useEffect as ie,useMemo as B,useState as oe}from"react";import G from"react";var re=({includeAllSections:r}={})=>{let{currentQuestionIndex:e,solutions:t,currentSectionIndex:o,test:i,isSolution:u}=D(),[n,T]=G.useState({MARKED_FOR_REVIEW:{count:0},ATTEMPTED:{count:0},ATTEMPTED_AND_MARKED_FOR_REVIEW:{count:0},NOT_ATTEMPTED:{count:0},NOT_VISITED:{count:0},ACITVE_QUESTION:{count:0}}),l=()=>{let s={...n};Object.values(s).forEach(E=>E.count=0),r?i?.sections?.forEach((E,a)=>{E?.questions?.forEach((f,U)=>{let S=u||i.type==="QUIZ"?I(U,a):m(U,a);s[S].count+=1})}):i?.sections[o]?.questions.forEach((E,a)=>{let f=u||i.type==="QUIZ"?I(a,o):m(a,o);s[f].count+=1}),T(s)};G.useEffect(()=>{l()},[e,t,o]);let m=(s,E)=>{let a=i?.sections[E]?.questions[s],f;return t[a?._id]&&t[a?._id].markedSolution?.length&&t[a?._id].markedForReview?f="ATTEMPTED_AND_MARKED_FOR_REVIEW":t[a?._id]&&t[a?._id].markedForReview?f="MARKED_FOR_REVIEW":t[a?._id]&&t[a?._id].markedSolution?.length?f="ATTEMPTED":t[a?._id]?f="NOT_ATTEMPTED":f="NOT_VISITED",f};function d(s,E){let a=s.slice().sort(),f=E.slice().sort();if(a.length!==f.length)return!1;for(let U=0;U<a.length;U++)if(a[U]!==f[U])return!1;return!0}let I=(s,E)=>{let a=i?.sections[E]?.questions[s],f;return t[a?._id]&&t[a?._id].markedSolution?.length?d(t[a?._id].markedSolution,a.solution)?f="ATTEMPTED":f="NOT_ATTEMPTED":f="NOT_VISITED",f};return{getQuestionStatus:m,questionStatusMap:n,getAnsweredQuestionStatus:I,areArraysEqual:d}},Q=re;var ue=()=>{let{handleUpdateQuestionCoordinates:r,currentQuestionIndex:e,currentSectionIndex:t,currentSection:o,currentQuestionAnswer:i,handleUpdateSolutions:u,solutions:n,currentQuestion:T,handleUpdateCurrentAnswer:l,test:m,userSettings:d}=D(),{getAnsweredQuestionStatus:I}=Q(),[s,E]=oe([]);ie(()=>{if(d?.selectedFilter){let _=o?.questions?.map((g,p)=>{if(d?.selectedFilter===I(p,t))return p})?.filter(g=>g!==void 0);E(_),_?.length&&r({currentSectionIndex:t,currentQuestionIndex:_[0]||0})}else E([])},[d?.selectedFilter,o]);let a=()=>{i?.isSaved||(delete i?.isSaved,u({...n,[T?._id]:{...i}}))},f=()=>{if(d?.selectedFilter){let p=s?.findIndex(y=>y===e);p>=0&&s[p+1]>=0&&r({currentSectionIndex:t,currentQuestionIndex:s[p+1]});return}let _=o?.questions[e+1]?t:m?.sections[t+1]?t+1:t,g=o?.questions[e+1]?e+1:0;r({currentSectionIndex:_,currentQuestionIndex:g})},U=()=>{if(d?.selectedFilter){let p=s?.findIndex(y=>y===e);p>=0&&s[p-1]>=0&&r({currentSectionIndex:t,currentQuestionIndex:s[p-1]});return}let _=o?.questions[e-1]?t:m?.sections[t-1]?t-1:t,g=o?.questions[e-1]?e-1:m.sections[_]?.questions?.length-1;r({currentSectionIndex:_,currentQuestionIndex:g})},S=B(()=>{if(d?.selectedFilter){let _=s?.findIndex(g=>g===e);return!s[_+1]}return!m?.sections[t+1]&&!o?.questions[e+1]},[t,e,s,d?.selectedFilter]),v=B(()=>d?.selectedFilter?s?.findIndex(g=>g===e)<=0:t===0&&e===0,[t,e,s,d?.selectedFilter]);return{handleNext:f,handlePrevious:U,handleSaveAnswer:a,handleMarkForReview:()=>{l({...i,markedForReview:!i?.markedForReview}),u({...n,[T._id]:{...n[T._id],markedForReview:!i?.markedForReview}})},isLastQuestion:S,isFirstQuestion:v}},ae=ue;import $ from"react";var K={1:{label:"A"},2:{label:"B"},3:{label:"C"},4:{label:"D"}},ce=()=>{let{currentQuestionIndex:r,test:e,currentSection:t,currentSectionIndex:o,currentQuestion:i}=D(),{getAnsweredQuestionStatus:u}=Q(),n=$.useMemo(()=>u(r,o)==="ATTEMPTED"?"correct":u(r,o)==="NOT_ATTEMPTED"?"incorrect":"skipped",[r,o]),T=$.useMemo(()=>{switch(n){case"correct":return e?.marksLevel==="TEST"?"+"+e.positiveMarks:"+"+t?.positiveMarks;case"incorrect":return e?.marksLevel==="TEST"?e.isNegativeMarkingEnabled?"-"+e.negativeMarks:0:t?.isNegativeMarkingEnabled?"-"+t?.negativeMarks:0;case"skipped":return 0}},[n,o,r]);return{correctAnswer:$.useMemo(()=>{if(!i)return"";if(i.type==="INTEGER")return i.solution[0];if(i.type==="SINGLE"){let m=i.options.findIndex(d=>d._id===i.solution[0]);return K[m+1]?.label||""}return i.type==="MULTIPLE"?i.solution.map(d=>{let I=i.options.findIndex(s=>s._id===d);return K[I+1]?.label||""}).join(", "):""},[i]),currentAnswerMarks:T,currentQuestionStatus:n}},le=ce;import V from"react";var de=r=>{let{test:e,handleSubmitTest:t}=D(),o=V.useRef(null),[i,u]=V.useState(e?.timeTaken?.[e?._id]??e?.duration??0),n=async l=>{let m=await c.getItem(`${e?._id}-timers`);if(l<=0){t(!1,!0);return}m[e._id]=l+"",c.setItem(`${e?._id}-timers`,m),u(l),o.current=setTimeout(()=>{n(l-1)},r==="app"?950:1e3)},T=async()=>{let l=await c.getItem(`${e?._id}-timers`);l||(await c.setItem(`${e?._id}-timers`,{}),l={});let m=parseInt(l?.[e?._id]||e?.timeTaken?.[e?._id]||e?.duration||0);n(m)};return V.useEffect(()=>(T(),()=>{o.current&&clearTimeout(o.current)}),[]),{timer:i}},me=de;import h from"react";var Te=r=>{let e=r==="app"?950:1e3,{currentQuestion:t,solutions:o,test:i}=D(),{isQuizAnswer:u}=L(),n=h.useRef(null),[T,l]=h.useState(0),m=async I=>{if(!t)return;let s=await c.getItem(`${i?._id}-timers`);if((o[t?._id]?.markedSolution||[])?.length===0)s[t?._id]=I+"",c.setItem(`${i?._id}-timers`,s),l(I);else{clearTimeout(n.current);return}n.current=setTimeout(()=>{m(I+1)},e)},d=async()=>{n.current&&(clearTimeout(n.current),n.current=null);let I=await c.getItem(`${i?._id}-timers`);I||(await c.setItem(`${i?._id}-timers`,{}),I={});let s=parseInt(I?.[t?._id]||i?.timeTaken?.[t?._id]||0);l(s),m(s)};return h.useEffect(()=>(t?._id&&d(),()=>{n.current&&(clearTimeout(n.current),n.current=null)}),[t]),h.useEffect(()=>{n.current===null&&setTimeout(()=>{n.current===null&&d()},e)},[o]),h.useEffect(()=>{u&&(clearTimeout(n.current),n.current=null)},[u]),{questionStopWatch:T}},Se=Te;export{w as AttemptStatus,H as TestEngineProvider,Se as useQuestionStopWatch,D as useTestEngine,L as useTestEngineBody,ae as useTestEngineFooter,Q as useTestQuestionStatus,le as useTestSolutionBody,me as useTestTimer};
1
+ var Z=(t=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(t,{get:(e,r)=>(typeof require<"u"?require:e)[r]}):t)(function(t){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+t+'" is not supported')});import ne,{useMemo as q}from"react";import{create as J}from"zustand";import{persist as X,createJSONStorage as ee}from"zustand/middleware";var $=()=>typeof globalThis>"u"||!globalThis.localStorage?null:{setItem:(t,e)=>{globalThis.localStorage.setItem(t,e)},getItem:t=>globalThis.localStorage.getItem(t)??null,removeItem:t=>{globalThis.localStorage.removeItem(t)}},Y=()=>{try{let{createMMKV:t}=Z("react-native-mmkv"),e=t();return{setItem:(r,s)=>{e.set(r,s)},getItem:r=>e.getString(r)??null,removeItem:r=>{e.delete(r)}}}catch{return null}},N={},j=()=>({setItem:(t,e)=>{N[t]=e},getItem:t=>N[t]??null,removeItem:t=>{delete N[t]}}),Q=$()||Y()||j();import te from"react";var P={test:void 0,solutions:{},currentQuestionIndex:0,currentSectionIndex:0,currentQuestionAnswer:{},userSettings:{},isSolution:!1,submitting:!1,timer:{},isHydrated:!1},y=te.createContext([P,()=>{}]);y.displayName="TestEngineContext";var o=J()(X(t=>({...P,setQuestionCoordinates:(e,r)=>t(s=>{let n=s.test?.sections[e]?.questions[r]?._id,c=null;return s.isSolution||(s.solutions[n]?c={...s.solutions,[n]:{...s.solutions[n],timeTaken:s.timer[n]||0}}:c={...s.solutions,[n]:{}}),{currentSectionIndex:e,currentQuestionIndex:r,currentQuestionAnswer:c?.[n]||{},...c&&{solutions:c}}}),setTest:(e,r)=>t(()=>{if(!r)return{test:e};let s={},u={},i=0,n=0;e?.sections?.forEach((E,p)=>{E?.questions?.forEach((T,R)=>{T?.questionResponse&&(s={...s,[T?._id]:{markedSolution:T?.questionResponse?.markedSolution||[],markedForReview:T?.questionResponse?.markedForReview||!1,timeTaken:T?.questionResponse?.timeTaken||0,isSaved:!0}},u={...u,[T?._id]:T?.questionResponse?.timeTaken||0}),e?.status==="Paused"&&T._id===e?.lastVisitedQuestionId&&(i=p,n=R)})}),u[e?._id]=e?.timeTaken&&+e?.timeTaken[e?._id]||+e?.duration||0;let c=e?.sections[i]?.questions[n]?._id,m=s[c]||{};return{test:e,solutions:s,timer:u,currentSectionIndex:i,currentQuestionIndex:n,currentQuestionAnswer:m}}),setSolution:(e,r)=>t(s=>({solutions:{...s.solutions,[e]:{...s.solutions[e],...r}},currentQuestionAnswer:{...s.solutions[e],...r}})),incrementTimer:e=>t(r=>({timer:{...r.timer,[e]:(r.timer[e]||0)+1}})),decrementTimer:e=>t(r=>({timer:{...r.timer,[e]:(r.timer[e]||0)-1}})),setCurrentAnswer:e=>t({currentQuestionAnswer:e}),setUserSettings:e=>t({userSettings:e}),setTestSubmitting:e=>t({submitting:e}),reset:()=>t(P),setIsHydrated:e=>t({isHydrated:e})}),{name:"test-engine-v2-store",storage:ee(()=>Q)}));var re=()=>{let t=ne.useContext(y);if(t===void 0)throw new Error("useTestEngineV2 must be use within TestEngineProvider");let[e,r]=t;if(!e.test)throw new Error("Please provide test data to useTestEngineV2");let s=o(S=>S.test),u=o(S=>S.setTestSubmitting),i=o(S=>S.solutions),n=o(S=>S.currentQuestionIndex),c=o(S=>S.currentSectionIndex),m=q(()=>s?.sections[c],[s,c]),E=q(()=>m?.questions[n],[m,n]),p=o(S=>S.reset),T=o(S=>S.timer);return{handleSubmitTest:(S=!1,I=!1)=>{u(!0);let A=[];s?.sections?.forEach(f=>{f?.questions?.forEach(d=>{i[d._id]&&A.push({questionId:d._id,markedSolution:i[d._id]?.isSaved?i[d._id]?.markedSolution:[],timeTaken:T[d._id]||i[d._id]?.timeTaken||0,markedForReview:i[d._id]?.markedForReview||!1})})});let k={submittedBy:I?"autoSubmit":"user",questionResponses:A,testMappingId:s?.testMappingId,type:S?"Pause":"Submit",testId:s?._id,...S?{lastVisitedQuestionId:E?._id}:{},...s?.type==="TEST"&&S&&T[s?._id]?{timeTaken:{[s?._id]:T[s?._id]}}:{}};return A.length||delete k.questionResponses,u(!1),r({payload:k,pause:S,autoSubmit:I}),k},currentSection:m,currentQuestion:E,...o.getState(),...e}},D=re;import{useEffect as L,useRef as se}from"react";var oe=()=>{let{isSolution:t,handleSubmitTest:e}=D(),r=o(m=>m.test),s=o(m=>m.timer),u=o(m=>m.decrementTimer),i=o(m=>m.isHydrated),n=se(null),c=()=>{r&&i&&u(r?._id),n.current=setTimeout(()=>{c()},1e3)};return L(()=>{if(r&&s[r?._id]<=0){clearTimeout(n.current),e(!1,!0);return}},[s,r]),L(()=>{if(!t)return c(),()=>{n.current&&clearTimeout(n.current)}},[r,i]),{timer:s[r._id]||0}},ie=oe;import O,{useCallback as V}from"react";var C=(n=>(n.ATTEMPTED="ATTEMPTED",n.NOT_ATTEMPTED="NOT_ATTEMPTED",n.NOT_VISITED="NOT_VISITED",n.MARKED_FOR_REVIEW="MARKED_FOR_REVIEW",n.ATTEMPTED_AND_MARKED_FOR_REVIEW="ATTEMPTED_AND_MARKED_FOR_REVIEW",n.ACITVE_QUESTION="ACITVE_QUESTION",n))(C||{});var K={1:{label:"A"},2:{label:"B"},3:{label:"C"},4:{label:"D"}},ae=({includeAllSections:t}={})=>{let e=o(f=>f.test),r=o(f=>f.solutions),s=o(f=>f.isSolution),u=o(f=>f.currentSectionIndex),{currentQuestion:i,currentSection:n}=D(),c=o(f=>f.currentQuestionIndex),[m,E]=O.useState({MARKED_FOR_REVIEW:{count:0},ATTEMPTED:{count:0},ATTEMPTED_AND_MARKED_FOR_REVIEW:{count:0},NOT_ATTEMPTED:{count:0},NOT_VISITED:{count:0},ACITVE_QUESTION:{count:0}}),p=V((f,d)=>{let a=e?.sections[d]?.questions[f],g=r[a?._id];return g?g?.markedSolution&&g?.markedSolution?.length>0&&g?.markedForReview?"ATTEMPTED_AND_MARKED_FOR_REVIEW":g?.markedSolution&&g?.markedSolution?.length>0?"ATTEMPTED":g?.markedForReview?"MARKED_FOR_REVIEW":"NOT_ATTEMPTED":"NOT_VISITED"},[r,e]),T=V((f,d)=>{let a=e?.sections[d]?.questions[f],g=r[a?._id];return g?.markedSolution?.length?S(g.markedSolution,a.solution)?"ATTEMPTED":"NOT_ATTEMPTED":"NOT_VISITED"},[r,e]),R=V(()=>{let f={...m};Object.values(f).forEach(d=>d.count=0),t?e?.sections?.forEach((d,a)=>{d?.questions?.forEach((g,M)=>{let w=s||e.type==="QUIZ"?T(M,a):p(M,a);f[w].count+=1})}):e?.sections[u]?.questions.forEach((d,a)=>{let g=s||e.type==="QUIZ"?T(a,u):p(a,u);f[g].count+=1}),E(f)},[T,p,t,s,e,u,r]);O.useEffect(()=>{R()},[c,r,u]);function S(f,d){let a=f.slice().sort(),g=d.slice().sort();if(a.length!==g.length)return!1;for(let M=0;M<a.length;M++)if(a[M]!==g[M])return!1;return!0}let I=O.useMemo(()=>T(c,u)==="ATTEMPTED"?"correct":T(c,u)==="NOT_ATTEMPTED"?"incorrect":"skipped",[c,u,T]),A=O.useMemo(()=>{switch(I){case"correct":return e?.marksLevel==="TEST"?"+"+e.positiveMarks:"+"+n?.positiveMarks;case"incorrect":return e?.marksLevel==="TEST"?e.isNegativeMarkingEnabled?"-"+e.negativeMarks:0:n?.isNegativeMarkingEnabled?"-"+n?.negativeMarks:0;case"skipped":return 0}},[I]),k=O.useMemo(()=>{if(!i)return"";if(i.type==="INTEGER")return i.solution[0];if(i.type==="SINGLE"){let f=i.options.findIndex(d=>d._id===i.solution[0]);return K[f+1]?.label||""}return i.type==="MULTIPLE"?i.solution.map(d=>{let a=i.options.findIndex(g=>g._id===d);return K[a+1]?.label||""}).join(", "):""},[i]);return{getQuestionStatus:p,questionStatusMap:m,getAnsweredQuestionStatus:T,areArraysEqual:S,currentQuestionStatus:I,currentAnswerMarks:A,correctAnswer:k}},U=ae;import{useEffect as ce,useRef as de}from"react";var le=()=>{let{currentQuestion:t,isSolution:e}=D(),r=o(m=>m.timer),s=o(m=>m.incrementTimer),u=o(m=>m.solutions),i=o(m=>m.isHydrated),n=de(null),c=()=>{t&&!u[t?._id]?.markedSolution?.length&&i&&s(t?._id),n.current=setTimeout(()=>{c()},1e3)};return ce(()=>{if(!e)return c(),()=>{n.current&&clearTimeout(n.current)}},[t,u,i]),{timer:r[t._id]||0}},me=le;import Te,{useCallback as h,useEffect as Se,useMemo as W,useState as fe}from"react";var ge=()=>{let t=o(l=>l.test),e=o(l=>l.solutions),{currentQuestion:r,currentSection:s,currentQuestionAnswer:u}=D(),i=o(l=>l.currentQuestionIndex),n=o(l=>l.currentSectionIndex),c=o(l=>l.userSettings),m=o(l=>l.isSolution),E=o(l=>l.setQuestionCoordinates),p=o(l=>l.setSolution),T=o(l=>l.setCurrentAnswer),{getAnsweredQuestionStatus:R}=U(),S=Te.useMemo(()=>t?.type==="QUIZ"&&!!e[r?._id]?.markedSolution?.length,[t,e,r]),I=h(()=>{T({...u,markedSolution:[],isSaved:!1}),p(r?._id,{...u,markedSolution:[]})},[u,r,p]),A=h(l=>{if(!(m||S))if(u?.markedSolution?.includes(l)){let _=u.markedSolution.filter(b=>b!==l);if(_?.length===0)return I();T({...u,markedSolution:_,isSaved:!1});return}else if(r?.type==="SINGLE"){T({...u,markedSolution:[l],isSaved:!1});return}else{let _=u?.markedSolution?.length?u?.markedSolution:[];T({...u,markedSolution:[..._,l],isSaved:!1});return}},[m,S,r,u,T,I]),k=h(l=>{m||S||(T({...u,markedSolution:l?[l]:[],isSaved:!1}),l||I())},[m,S,u,I]),f=h(()=>{T({...u,markedForReview:!u?.markedForReview}),p(r._id,{...e[r._id],markedForReview:!u?.markedForReview})},[p,u,e,r]),[d,a]=fe([]);Se(()=>{if(c?.selectedFilter){let l=s?.questions?.map((_,b)=>{if(c?.selectedFilter===R(b,n))return b})?.filter(_=>_!==void 0);a(l),l?.length&&E(n,l[0]||0)}else a([])},[c,s,R,E]);let g=h(()=>{u?.isSaved||p(r?._id,{...u,isSaved:!0})},[u,r,p]),M=h(()=>{if(c?.selectedFilter){let b=d?.findIndex(F=>F===i);b>=0&&d[b+1]>=0&&E(n,d[b+1]);return}let l=s?.questions[i+1]?n:t?.sections[n+1]?n+1:n,_=s?.questions[i+1]?i+1:0;E(l,_)},[c?.selectedFilter,d,i,n,s,t,E]),w=h(()=>{if(c?.selectedFilter){let b=d?.findIndex(F=>F===i);b>=0&&d[b-1]>=0&&E(n,d[b-1]);return}let l=s?.questions[i-1]?n:t?.sections[n-1]?n-1:n,_=s?.questions[i-1]?i-1:t.sections[l]?.questions?.length-1;E(l,_)},[c?.selectedFilter,d,i,n,s,t,E]),z=W(()=>{if(c?.selectedFilter){let l=d?.findIndex(_=>_===i);return!d[l+1]}return!t?.sections[n+1]&&!s?.questions[i+1]},[n,i,d,c?.selectedFilter,t,s]),B=W(()=>c?.selectedFilter?d?.findIndex(_=>_===i)<=0:n===0&&i===0,[n,i,d,c?.selectedFilter]);return{isQuizAnswer:S,handleClearResponse:I,handleSelectOption:A,handleIntegerInputChange:k,handleMarkForReview:f,handleNext:M,handlePrevious:w,handleSaveAnswer:g,isLastQuestion:z,isFirstQuestion:B}},Ee=ge;import{useEffect as pe,forwardRef as Ie,useImperativeHandle as _e}from"react";import G from"react";var be=Ie(({test:t,isSolution:e,submitTest:r,children:s},u)=>{let i=o(a=>a.setTest),n=o(a=>a.setIsHydrated),c=o(a=>a.isHydrated),m=o(a=>a.test),E=o(a=>a.solutions),p=o(a=>a.currentQuestionIndex),T=o(a=>a.currentSectionIndex),R=o(a=>a.currentQuestionAnswer),S=o(a=>a.userSettings),I=o(a=>a.submitting),A=o(a=>a.timer),k=o(a=>a.reset),f=G.useMemo(()=>({test:m,solutions:E,currentQuestionIndex:p,currentSectionIndex:T,userSettings:S,isSolution:e||!1,submitting:I,timer:A,isHydrated:c,currentQuestionAnswer:R}),[m,E,p,T,S,e,I,A,c,R]),d=()=>{console.log("[RESET]: Called"),k(),Q.removeItem(`${t?._id}-test-initiated`),Q.removeItem(`test-engine-v2-store-${t?._id}`),Q.removeItem("test-engine-v2-store")};return _e(u,()=>({reset:d})),pe(()=>{(async()=>{if(!(c&&t?._id===m?._id))try{Q.getItem(`${t?._id}-test-initiated`)?i(t,!1):(i(t,!0),e||Q.setItem(`${t?._id}-test-initiated`,"true")),o.persist.setOptions({name:`test-engine-v2-store-${t?._id}`}),await o.persist.rehydrate(),Q.removeItem("test-engine-v2-store"),n(!0)}catch(g){console.error("Error initializing test engine store:",g),n(!0)}})()},[t?._id,c,i,n]),!c||!m?null:G.createElement(y.Provider,{value:[f,r]},s)}),Re=be;export{C as AttemptStatus,Re as TestEngineProvider,me as useQuestionTimer,U as useQuestions,Ee as useTestActions,D as useTestEngine,ie as useTestTimer};
package/package.json CHANGED
@@ -1,28 +1,32 @@
1
1
  {
2
2
  "name": "testforce-engine",
3
- "version": "0.0.3",
3
+ "version": "1.0.0-beta-1",
4
4
  "description": "",
5
5
  "main": "dist/index.js",
6
6
  "module": "dist/index.mjs",
7
7
  "types": "dist/index.d.ts",
8
8
  "scripts": {
9
9
  "build": "tsup",
10
+ "build:watch": "tsup --watch",
10
11
  "prepare": "npm run build"
11
12
  },
12
13
  "keywords": [],
13
14
  "author": "",
14
15
  "license": "ISC",
15
16
  "dependencies": {
16
- "@babel/runtime": "^7.26.9",
17
- "@react-native-async-storage/async-storage": "^2.1.1"
17
+ "@babel/runtime": "^7.26.9"
18
18
  },
19
19
  "devDependencies": {
20
- "@types/react": "^19.0.8",
20
+ "@types/react": "18.3.1",
21
+ "react-native-mmkv": "^4.1.1",
21
22
  "tsup": "^8.3.6",
22
- "typescript": "^5.7.3"
23
+ "typescript": "^5.7.3",
24
+ "zustand": "^5.0.10"
23
25
  },
24
26
  "peerDependencies": {
25
- "react": ">=18",
26
- "react-native": ">=0.72"
27
+ "react": "18.3.1",
28
+ "react-native": "0.76.9",
29
+ "react-native-mmkv": ">=4",
30
+ "zustand": ">=5"
27
31
  }
28
32
  }