module GenerateurH5P exposing (..) import Array as A import Browser exposing (Document) import Dict import Element exposing (..) import Element.Background as Background import Element.Border as Border import Element.Events exposing (..) import Element.Font as Font import Element.Input as Input import File.Download import Html exposing (Attribute, Html, button, div, iframe, input, p, section, textarea) import Json.Decode as D import Json.Encode as E import List as L import Parser.Advanced as P exposing (..) import Random import Random.Extra import Random.List import Set import String as S import Style exposing (..) import Tuple import UUID exposing (UUID) titre = "Générateur d'archives H5P" {- ███ ███ ██████ ██████ ███████ ██ ████ ████ ██ ██ ██ ██ ██ ██ ██ ████ ██ ██ ██ ██ ██ █████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██████ ███████ ███████ -} type alias Model = { structureDuContenu : String , contenuGenere : String } init : Model init = { structureDuContenu = "" , contenuGenere = "" } {- ██ ██ ██████ ██████ █████ ████████ ███████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██ ██ ███████ ██ █████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██ ██████ ██ ██ ██ ███████ -} type Msg = StructureDuContenu String | GenererContenu | TelechargerContenu update : Msg -> Model -> ( Model, Cmd Msg ) update msg model = case msg of StructureDuContenu nouvelleStructure -> let f strCtn = case P.run parser strCtn of Ok ctn -> S.join "\n\n" <| L.map (h5pEncode 2) ctn Err erreurs -> deadEndsToStringBis erreurs in ( { model | structureDuContenu = nouvelleStructure , contenuGenere = f model.structureDuContenu } , Cmd.none ) GenererContenu -> let f strCtn = case P.run parser strCtn of Ok ctn -> S.join "\n\n" <| L.map (h5pEncode 0) ctn Err erreurs -> deadEndsToStringBis erreurs in ( { model | contenuGenere = f model.structureDuContenu } , Cmd.none ) TelechargerContenu -> ( model , File.Download.string "content.json" "text/json" model.contenuGenere ) -- SUBSCRIPTIONS subscriptions : Model -> Sub Msg subscriptions model = Sub.none {- ██ ██ ██ ███████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ █████ ██ █ ██ ██ ██ ██ ██ ██ ███ ██ ████ ██ ███████ ███ ███ -} view : Model -> Element Msg view model = row [ spacing grandEspacement , padding tresGrandEspacement , height fill , width fill , scrollbars ] [ Input.multiline [ height fill , width fill , clip , scrollbars , Background.color <| couleurUI <| couleurArrierePlan , Border.rounded 8 , Border.innerShadow { blur = 10 , color = rgb255 10 10 10 , offset = ( 0.3, 0.4 ) , size = 2 } ] { onChange = StructureDuContenu , label = Input.labelHidden "chose" , placeholder = Just <| Input.placeholder [] <| text "Structure du contenu" , text = model.structureDuContenu , spellcheck = True } , column [ spacing petitEspacement, height fill, width fill, scrollbars ] -- L'attibut scrollbars présent dans la liste ci-dessus ^^^^^^^^^^ -- est nécessaire pour que l'élément ci-dessous ne s'étende pas ! [ row [ width fill , padding petitEspacement , spacing tresGrandEspacement ] [ bouton GenererContenu "Générer le contenu" , bouton TelechargerContenu "Télécharger" ] , el --^^ Cet élément ci [ height fill , width fill , clip , scrollbars --^^^^^^^^^^ Cet attribut ne suffit pas , padding petitEspacement , Background.color <| couleurUI <| couleurArrierePlan , Border.rounded 8 , Border.innerShadow { blur = 10 , color = rgb255 10 10 10 , offset = ( 0.3, 0.4 ) , size = 2 } ] <| text model.contenuGenere ] ] source = """{ "media": { "disableImageZooming": false }, "correct": "true", "behaviour": { "enableRetry": true, "enableSolutionsButton": true, "enableCheckButton": true, "confirmCheckDialog": false, "confirmRetryDialog": false, "autoCheck": true, "feedbackOnCorrect": "C'est la base !\\n" }, "l10n": { "trueText": "Vrai", "falseText": "Faux", "score": "Vous avez obtenu @score points sur un total de @total", "checkAnswer": "Vérifier", "submitAnswer": "Vérifier", "showSolutionButton": "Voir la solution", "tryAgain": "Recommencer", "wrongAnswerMessage": "Réponse incorrecte", "correctAnswerMessage": "Bonne réponse", "scoreBarLabel": "Vous avez obtenu @score points sur un total de @total", "a11yCheck": "Check the answers. The responses will be marked as correct, incorrect, or unanswered.", "a11yShowSolution": "Show the solution. The task will be marked with its correct solution.", "a11yRetry": "Retry the task. Reset all responses and start the task over again." }, "confirmCheck": { "header": "Terminer ?", "body": "Êtes-vous sûr de vouloir terminer ?", "cancelLabel": "Annuler", "confirmLabel": "Confirmer" }, "confirmRetry": { "header": "Recommencer ?", "body": "Êtes-vous sûr de vouloir recommencer ?", "cancelLabel": "Annuler", "confirmLabel": "Confirmer" }, "question": "
Est-ce que \\\\(2+2=4\\\\) ?
\\n" }""" {- .--..--..--..--..--..--..--..--..--..--..--..--..--..--..--..--. / .. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \ \ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/ / \/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ / / /\/ /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /\/ /\ / /\ \/`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'\ \/\ \ \ \/\ \ /\ \/ / \/ /\ \ ██╗ ██╗███████╗██████╗ / /\/ / / /\/ / ██║ ██║██╔════╝██╔══██╗ \ \/ /\ / /\ \/ ███████║███████╗██████╔╝ \ \/\ \ \ \/\ \ ██╔══██║╚════██║██╔═══╝ /\ \/ / \/ /\ \ ██║ ██║███████║██║ / /\/ / / /\/ / ╚═╝ ╚═╝╚══════╝╚═╝ \ \/ /\ / /\ \/ \ \/\ \ \ \/\ \.--..--..--..--..--..--..--..--..--..--..--..--..--./\ \/ / \/ /\/ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ /\/ / / /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\ / /\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \ \ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `' / `--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--' -} type H5P = EmptyH5p | BranchingScenarioH5P BranchingScenario | CoursePresentationH5P CoursePresentation | TrueFalseH5P TrueFalse h5pEncode indent content = E.encode indent <| case content of EmptyH5p -> E.object [] BranchingScenarioH5P branchingScenario -> encodedBranchingScenario branchingScenario CoursePresentationH5P coursePresentation -> encodedCoursePresentation coursePresentation TrueFalseH5P trueFalse -> encodedTrueFalse trueFalse {- ██████ ██████ █████ ███ ██ ██████ ██ ██ ██ ███ ██ ██████ ██ ██ ██ ██ ██ ██ ████ ██ ██ ██ ██ ██ ████ ██ ██ ██████ ██████ ███████ ██ ██ ██ ██ ███████ ██ ██ ██ ██ ██ ███ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██ ██ ██ ██ ██ ████ ██████ ██ ██ ██ ██ ████ ██████ ███████ ██████ ███████ ███ ██ █████ ██████ ██ ██████ ██ ██ ██ ████ ██ ██ ██ ██ ██ ██ ██ ██ ███████ ██ █████ ██ ██ ██ ███████ ██████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ███████ ██████ ███████ ██ ████ ██ ██ ██ ██ ██ ██████ -} type alias BranchingScenario = { behaviour : BranchingScenarioBehaviour , endScreens : List BranchingScenarioEndScreensObject , l10n : BranchingScenarioL10n , scoringOptionGroup : BranchingScenarioScoringOptionGroup , startScreen : BranchingScenarioStartScreen , content : List BranchingScenarioContent } type alias BranchingScenarioBehaviour = { enableBackwardsNavigation : Bool , forceContentFinished : Bool } type alias BranchingScenarioEndScreensObject = { contentId : Int , endScreenScore : Int , endScreenSubtitle : String , endScreenTitle : String } type alias BranchingScenarioL10n = { backButtonText : String , disableProceedButtonText : String , endScreenButtonText : String , fullscreenAria : String , proceedButtonText : String , replayButtonText : String , scoreText : String , startScreenButtonText : String } type alias BranchingScenarioScoringOptionGroup = { includeInteractionsScores : Bool , scoringOption : String } type alias BranchingScenarioStartScreen = { startScreenSubtitle : String , startScreenTitle : String } type alias BranchingScenarioContent = { contentBehaviour : String , feedback : BranchingScenarioContentFeedback , forceContentFinished : String , showContentTitle : Bool , type_ : BranchingScenarioContentType } type alias BranchingScenarioContentFeedback = { subtitle : String } type alias BranchingScenarioContentType = { library : String , metadata : BranchingScenarioContentTypeMetadata , params : BranchingScenarioContentTypeParams , subContentId : String } type alias BranchingScenarioContentTypeMetadata = { contentType : String , license : String , title : String } type alias BranchingScenarioContentTypeParams = H5P branchingScenarioDecoder : D.Decoder BranchingScenario branchingScenarioDecoder = D.map6 BranchingScenario (D.field "behaviour" branchingScenarioBehaviourDecoder) (D.field "endScreens" <| D.list branchingScenarioEndScreensObjectDecoder) (D.field "l10n" branchingScenarioL10nDecoder) (D.field "scoringOptionGroup" branchingScenarioScoringOptionGroupDecoder) (D.field "startScreen" branchingScenarioStartScreenDecoder) (D.field "content" <| D.list branchingScenarioContentDecoder) branchingScenarioBehaviourDecoder : D.Decoder BranchingScenarioBehaviour branchingScenarioBehaviourDecoder = D.map2 BranchingScenarioBehaviour (D.field "enableBackwardsNavigation" D.bool) (D.field "forceContentFinished" D.bool) branchingScenarioEndScreensObjectDecoder : D.Decoder BranchingScenarioEndScreensObject branchingScenarioEndScreensObjectDecoder = D.map4 BranchingScenarioEndScreensObject (D.field "contentId" D.int) (D.field "endScreenScore" D.int) (D.field "endScreenSubtitle" D.string) (D.field "endScreenTitle" D.string) branchingScenarioL10nDecoder : D.Decoder BranchingScenarioL10n branchingScenarioL10nDecoder = D.map8 BranchingScenarioL10n (D.field "backButtonText" D.string) (D.field "disableProceedButtonText" D.string) (D.field "endScreenButtonText" D.string) (D.field "fullscreenAria" D.string) (D.field "proceedButtonText" D.string) (D.field "replayButtonText" D.string) (D.field "scoreText" D.string) (D.field "startScreenButtonText" D.string) branchingScenarioScoringOptionGroupDecoder : D.Decoder BranchingScenarioScoringOptionGroup branchingScenarioScoringOptionGroupDecoder = D.map2 BranchingScenarioScoringOptionGroup (D.field "includeInteractionsScores" D.bool) (D.field "scoringOption" D.string) branchingScenarioStartScreenDecoder : D.Decoder BranchingScenarioStartScreen branchingScenarioStartScreenDecoder = D.map2 BranchingScenarioStartScreen (D.field "startScreenSubtitle" D.string) (D.field "startScreenTitle" D.string) branchingScenarioContentDecoder : D.Decoder BranchingScenarioContent branchingScenarioContentDecoder = D.map5 BranchingScenarioContent (D.field "contentBehaviour" D.string) (D.field "feedback" branchingScenarioContentFeedbackDecoder) (D.field "forceContentFinished" D.string) (D.field "showContentTitle" D.bool) (D.field "type" branchingScenarioContentTypeDecoder) branchingScenarioContentFeedbackDecoder : D.Decoder BranchingScenarioContentFeedback branchingScenarioContentFeedbackDecoder = D.map BranchingScenarioContentFeedback (D.field "subtitle" D.string) branchingScenarioContentTypeDecoder : D.Decoder BranchingScenarioContentType branchingScenarioContentTypeDecoder = D.map4 BranchingScenarioContentType (D.field "library" D.string) (D.field "metadata" branchingScenarioContentTypeMetadataDecoder) (D.field "params" branchingScenarioContentTypeParamsDecoder) (D.field "subContentId" D.string) branchingScenarioContentTypeMetadataDecoder : D.Decoder BranchingScenarioContentTypeMetadata branchingScenarioContentTypeMetadataDecoder = D.map3 BranchingScenarioContentTypeMetadata (D.field "contentType" D.string) (D.field "license" D.string) (D.field "title" D.string) branchingScenarioContentTypeParamsDecoder : D.Decoder BranchingScenarioContentTypeParams branchingScenarioContentTypeParamsDecoder = D.succeed EmptyH5p encodedBranchingScenario : BranchingScenario -> E.Value encodedBranchingScenario branchingScenario = E.object [ ( "branchingScenario" , E.object [ ( "behaviour", encodedBranchingScenarioBehaviour branchingScenario.behaviour ) , ( "endScreens", E.list encodedBranchingScenarioEndScreensObject branchingScenario.endScreens ) , ( "l10n", encodedBranchingScenarioL10n branchingScenario.l10n ) , ( "scoringOptionGroup", encodedBranchingScenarioScoringOptionGroup branchingScenario.scoringOptionGroup ) , ( "startScreen", encodedBranchingScenarioStartScreen branchingScenario.startScreen ) ] ) ] encodedBranchingScenarioBehaviour : BranchingScenarioBehaviour -> E.Value encodedBranchingScenarioBehaviour branchingScenarioBehaviour = E.object [ ( "enableBackwardsNavigation", E.bool branchingScenarioBehaviour.enableBackwardsNavigation ) , ( "forceContentFinished", E.bool branchingScenarioBehaviour.forceContentFinished ) ] encodedBranchingScenarioEndScreensObject : BranchingScenarioEndScreensObject -> E.Value encodedBranchingScenarioEndScreensObject branchingScenarioEndScreensObject = E.object [ ( "contentId", E.int branchingScenarioEndScreensObject.contentId ) , ( "endScreenScore", E.int branchingScenarioEndScreensObject.endScreenScore ) , ( "endScreenSubtitle", E.string branchingScenarioEndScreensObject.endScreenSubtitle ) , ( "endScreenTitle", E.string branchingScenarioEndScreensObject.endScreenTitle ) ] encodedBranchingScenarioL10n : BranchingScenarioL10n -> E.Value encodedBranchingScenarioL10n branchingScenarioL10n = E.object [ ( "backButtonText", E.string branchingScenarioL10n.backButtonText ) , ( "disableProceedButtonText", E.string branchingScenarioL10n.disableProceedButtonText ) , ( "endScreenButtonText", E.string branchingScenarioL10n.endScreenButtonText ) , ( "fullscreenAria", E.string branchingScenarioL10n.fullscreenAria ) , ( "proceedButtonText", E.string branchingScenarioL10n.proceedButtonText ) , ( "replayButtonText", E.string branchingScenarioL10n.replayButtonText ) , ( "scoreText", E.string branchingScenarioL10n.scoreText ) , ( "startScreenButtonText", E.string branchingScenarioL10n.startScreenButtonText ) ] encodedBranchingScenarioScoringOptionGroup : BranchingScenarioScoringOptionGroup -> E.Value encodedBranchingScenarioScoringOptionGroup branchingScenarioScoringOptionGroup = E.object [ ( "includeInteractionsScores", E.bool branchingScenarioScoringOptionGroup.includeInteractionsScores ) , ( "scoringOption", E.string branchingScenarioScoringOptionGroup.scoringOption ) ] encodedBranchingScenarioStartScreen : BranchingScenarioStartScreen -> E.Value encodedBranchingScenarioStartScreen branchingScenarioStartScreen = E.object [ ( "startScreenSubtitle", E.string branchingScenarioStartScreen.startScreenSubtitle ) , ( "startScreenTitle", E.string branchingScenarioStartScreen.startScreenTitle ) ] encodedBranchingScenarioContent : BranchingScenarioContent -> E.Value encodedBranchingScenarioContent branchingScenarioContent = E.object [ ( "contentBehaviour", E.string branchingScenarioContent.contentBehaviour ) , ( "feedback", encodedBranchingScenarioContentFeedback branchingScenarioContent.feedback ) , ( "forceContentFinished", E.string branchingScenarioContent.forceContentFinished ) , ( "showContentTitle", E.bool branchingScenarioContent.showContentTitle ) , ( "type", encodedBranchingScenarioContentType branchingScenarioContent.type_ ) ] encodedBranchingScenarioContentFeedback : BranchingScenarioContentFeedback -> E.Value encodedBranchingScenarioContentFeedback branchingScenarioContentFeedback = E.object [ ( "subtitle", E.string branchingScenarioContentFeedback.subtitle ) ] encodedBranchingScenarioContentType : BranchingScenarioContentType -> E.Value encodedBranchingScenarioContentType branchingScenarioContentType = E.object [ ( "library", E.string branchingScenarioContentType.library ) , ( "metadata", encodedBranchingScenarioContentTypeMetadata branchingScenarioContentType.metadata ) , ( "params", encodedBranchingScenarioContentTypeParams branchingScenarioContentType.params ) , ( "subContentId", E.string branchingScenarioContentType.subContentId ) ] encodedBranchingScenarioContentTypeMetadata : BranchingScenarioContentTypeMetadata -> E.Value encodedBranchingScenarioContentTypeMetadata branchingScenarioContentTypeMetadata = E.object [ ( "contentType", E.string branchingScenarioContentTypeMetadata.contentType ) , ( "license", E.string branchingScenarioContentTypeMetadata.license ) , ( "title", E.string branchingScenarioContentTypeMetadata.title ) ] encodedBranchingScenarioContentTypeParams : BranchingScenarioContentTypeParams -> E.Value encodedBranchingScenarioContentTypeParams branchingScenarioContentTypeParams = case branchingScenarioContentTypeParams of CoursePresentationH5P x -> encodedCoursePresentation x --TODO _ -> E.object [] nouveauBranchingScenario = { endScreens = [ { endScreenTitle = "Fin du parcours personnalisé" , endScreenSubtitle = "Fin du parcours personnalisé" , contentId = -1 , endScreenScore = 0 } ] , scoringOptionGroup = { scoringOption = "no-score" , includeInteractionsScores = True } , startScreen = { startScreenTitle = "Parcours personnalisé
\n" , startScreenSubtitle = "Préparez bien vos méninges !
\n" } , behaviour = { enableBackwardsNavigation = True , forceContentFinished = False } , l10n = { startScreenButtonText = "Commencer le parcours" , endScreenButtonText = "Recommencer le parcours" , backButtonText = "Revenir en arrière" , proceedButtonText = "Continuer" , disableProceedButtonText = "Jouer la vidéo de nouveau" , replayButtonText = "Votre note:" , scoreText = "Votre note:" , fullscreenAria = "Plein écran" } , content = [] } {- ██████ ██████ ██ ██ ██████ ███████ ███████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ███████ █████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██████ ██████ ██ ██ ███████ ███████ ██████ ██████ ███████ ███████ ███████ ███ ██ ████████ █████ ████████ ██ ██████ ███ ██ ██ ██ ██ ██ ██ ██ ██ ████ ██ ██ ██ ██ ██ ██ ██ ██ ████ ██ ██████ ██████ █████ ███████ █████ ██ ██ ██ ██ ███████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ███████ ███████ ███████ ██ ████ ██ ██ ██ ██ ██ ██████ ██ ████ -} type alias CoursePresentation = { l10n : CoursePresentationL10n , override : CoursePresentationOverride , presentation : CoursePresentationPresentation } type alias CoursePresentationL10n = { accessibilityCanvasLabel : String , accessibilityEnteredFullscreen : String , accessibilityExitedFullscreen : String , accessibilitySlideNavigationExplanation : String , accessibilityTotalScore : String , confirmDialogConfirmText : String , confirmDialogHeader : String , confirmDialogText : String , containsCompleted : String , containsIncorrectAnswers : String , containsNotCompleted : String , containsOnlyCorrect : String , currentSlide : String , exitFullscreen : String , exportAnswers : String , fullscreen : String , hideKeywords : String , lastSlide : String , maxScore : String , nextSlide : String , noTitle : String , prevSlide : String , printAllSlides : String , printCurrentSlide : String , printIngress : String , printTitle : String , retry : String , score : String , scoreMessage : String , shareFacebook : String , shareGoogle : String , shareResult : String , shareTwitter : String , showKeywords : String , showSolutions : String , slide : String , slideCount : String , solutionModeText : String , solutionModeTitle : String , solutionsButtonTitle : String , summary : String , summaryMultipleTaskText : String , total : String , totalScore : String , yourScore : String } type alias CoursePresentationOverride = { activeSurface : Bool , enablePrintButton : Bool , hideSummarySlide : Bool , social : CoursePresentationOverrideSocial , summarySlideRetryButton : Bool , summarySlideSolutionButton : Bool } type alias CoursePresentationOverrideSocial = { facebookShare : CoursePresentationOverrideSocialFacebookShare , googleShareUrl : String , showFacebookShare : Bool , showGoogleShare : Bool , showTwitterShare : Bool , twitterShare : CoursePresentationOverrideSocialTwitterShare } type alias CoursePresentationOverrideSocialFacebookShare = { quote : String , url : String } type alias CoursePresentationOverrideSocialTwitterShare = { hashtags : String , statement : String , url : String } type alias CoursePresentationPresentation = { globalBackgroundSelector : CoursePresentationPresentationGlobalBackgroundSelector , keywordListAlwaysShow : Bool , keywordListAutoHide : Bool , keywordListEnabled : Bool , keywordListOpacity : Int , slides : List CoursePresentationPresentationSlidesObject } type alias CoursePresentationPresentationGlobalBackgroundSelector = { fillGlobalBackground : String } type alias CoursePresentationPresentationSlidesObject = { elements : List () , slideBackgroundSelector : CoursePresentationPresentationSlidesObjectSlideBackgroundSelector } type alias CoursePresentationPresentationSlidesObjectSlideBackgroundSelector = { fillSlideBackground : String } coursePresentationDecoder : D.Decoder CoursePresentation coursePresentationDecoder = D.map3 CoursePresentation (D.field "l10n" coursePresentationL10nDecoder) (D.field "override" coursePresentationOverrideDecoder) (D.field "presentation" coursePresentationPresentationDecoder) coursePresentationL10nDecoder : D.Decoder CoursePresentationL10n coursePresentationL10nDecoder = let fieldSet0 = D.map8 CoursePresentationL10n (D.field "accessibilityCanvasLabel" D.string) (D.field "accessibilityEnteredFullscreen" D.string) (D.field "accessibilityExitedFullscreen" D.string) (D.field "accessibilitySlideNavigationExplanation" D.string) (D.field "accessibilityTotalScore" D.string) (D.field "confirmDialogConfirmText" D.string) (D.field "confirmDialogHeader" D.string) (D.field "confirmDialogText" D.string) fieldSet1 = D.map8 (<|) fieldSet0 (D.field "containsCompleted" D.string) (D.field "containsIncorrectAnswers" D.string) (D.field "containsNotCompleted" D.string) (D.field "containsOnlyCorrect" D.string) (D.field "currentSlide" D.string) (D.field "exitFullscreen" D.string) (D.field "exportAnswers" D.string) fieldSet2 = D.map8 (<|) fieldSet1 (D.field "fullscreen" D.string) (D.field "hideKeywords" D.string) (D.field "lastSlide" D.string) (D.field "maxScore" D.string) (D.field "nextSlide" D.string) (D.field "noTitle" D.string) (D.field "prevSlide" D.string) fieldSet3 = D.map8 (<|) fieldSet2 (D.field "printAllSlides" D.string) (D.field "printCurrentSlide" D.string) (D.field "printIngress" D.string) (D.field "printTitle" D.string) (D.field "retry" D.string) (D.field "score" D.string) (D.field "scoreMessage" D.string) fieldSet4 = D.map8 (<|) fieldSet3 (D.field "shareFacebook" D.string) (D.field "shareGoogle" D.string) (D.field "shareResult" D.string) (D.field "shareTwitter" D.string) (D.field "showKeywords" D.string) (D.field "showSolutions" D.string) (D.field "slide" D.string) fieldSet5 = D.map8 (<|) fieldSet4 (D.field "slideCount" D.string) (D.field "solutionModeText" D.string) (D.field "solutionModeTitle" D.string) (D.field "solutionsButtonTitle" D.string) (D.field "summary" D.string) (D.field "summaryMultipleTaskText" D.string) (D.field "total" D.string) in D.map3 (<|) fieldSet5 (D.field "totalScore" D.string) (D.field "yourScore" D.string) coursePresentationOverrideDecoder : D.Decoder CoursePresentationOverride coursePresentationOverrideDecoder = D.map6 CoursePresentationOverride (D.field "activeSurface" D.bool) (D.field "enablePrintButton" D.bool) (D.field "hideSummarySlide" D.bool) (D.field "social" coursePresentationOverrideSocialDecoder) (D.field "summarySlideRetryButton" D.bool) (D.field "summarySlideSolutionButton" D.bool) coursePresentationOverrideSocialDecoder : D.Decoder CoursePresentationOverrideSocial coursePresentationOverrideSocialDecoder = D.map6 CoursePresentationOverrideSocial (D.field "facebookShare" coursePresentationOverrideSocialFacebookShareDecoder) (D.field "googleShareUrl" D.string) (D.field "showFacebookShare" D.bool) (D.field "showGoogleShare" D.bool) (D.field "showTwitterShare" D.bool) (D.field "twitterShare" coursePresentationOverrideSocialTwitterShareDecoder) coursePresentationOverrideSocialFacebookShareDecoder : D.Decoder CoursePresentationOverrideSocialFacebookShare coursePresentationOverrideSocialFacebookShareDecoder = D.map2 CoursePresentationOverrideSocialFacebookShare (D.field "quote" D.string) (D.field "url" D.string) coursePresentationOverrideSocialTwitterShareDecoder : D.Decoder CoursePresentationOverrideSocialTwitterShare coursePresentationOverrideSocialTwitterShareDecoder = D.map3 CoursePresentationOverrideSocialTwitterShare (D.field "hashtags" D.string) (D.field "statement" D.string) (D.field "url" D.string) coursePresentationPresentationDecoder : D.Decoder CoursePresentationPresentation coursePresentationPresentationDecoder = D.map6 CoursePresentationPresentation (D.field "globalBackgroundSelector" coursePresentationPresentationGlobalBackgroundSelectorDecoder) (D.field "keywordListAlwaysShow" D.bool) (D.field "keywordListAutoHide" D.bool) (D.field "keywordListEnabled" D.bool) (D.field "keywordListOpacity" D.int) (D.field "slides" <| D.list coursePresentationPresentationSlidesObjectDecoder) coursePresentationPresentationGlobalBackgroundSelectorDecoder : D.Decoder CoursePresentationPresentationGlobalBackgroundSelector coursePresentationPresentationGlobalBackgroundSelectorDecoder = D.map CoursePresentationPresentationGlobalBackgroundSelector (D.field "fillGlobalBackground" D.string) coursePresentationPresentationSlidesObjectDecoder : D.Decoder CoursePresentationPresentationSlidesObject coursePresentationPresentationSlidesObjectDecoder = D.map2 CoursePresentationPresentationSlidesObject (D.field "elements" <| D.list <| D.succeed ()) (D.field "slideBackgroundSelector" coursePresentationPresentationSlidesObjectSlideBackgroundSelectorDecoder) coursePresentationPresentationSlidesObjectSlideBackgroundSelectorDecoder : D.Decoder CoursePresentationPresentationSlidesObjectSlideBackgroundSelector coursePresentationPresentationSlidesObjectSlideBackgroundSelectorDecoder = D.map CoursePresentationPresentationSlidesObjectSlideBackgroundSelector (D.field "fillSlideBackground" D.string) encodedCoursePresentation : CoursePresentation -> E.Value encodedCoursePresentation coursePresentation = E.object [ ( "l10n", encodedCoursePresentationL10n coursePresentation.l10n ) , ( "override", encodedCoursePresentationOverride coursePresentation.override ) , ( "presentation", encodedCoursePresentationPresentation coursePresentation.presentation ) ] encodedCoursePresentationL10n : CoursePresentationL10n -> E.Value encodedCoursePresentationL10n coursePresentationL10n = E.object [ ( "accessibilityCanvasLabel", E.string coursePresentationL10n.accessibilityCanvasLabel ) , ( "accessibilityEnteredFullscreen", E.string coursePresentationL10n.accessibilityEnteredFullscreen ) , ( "accessibilityExitedFullscreen", E.string coursePresentationL10n.accessibilityExitedFullscreen ) , ( "accessibilitySlideNavigationExplanation", E.string coursePresentationL10n.accessibilitySlideNavigationExplanation ) , ( "accessibilityTotalScore", E.string coursePresentationL10n.accessibilityTotalScore ) , ( "confirmDialogConfirmText", E.string coursePresentationL10n.confirmDialogConfirmText ) , ( "confirmDialogHeader", E.string coursePresentationL10n.confirmDialogHeader ) , ( "confirmDialogText", E.string coursePresentationL10n.confirmDialogText ) , ( "containsCompleted", E.string coursePresentationL10n.containsCompleted ) , ( "containsIncorrectAnswers", E.string coursePresentationL10n.containsIncorrectAnswers ) , ( "containsNotCompleted", E.string coursePresentationL10n.containsNotCompleted ) , ( "containsOnlyCorrect", E.string coursePresentationL10n.containsOnlyCorrect ) , ( "currentSlide", E.string coursePresentationL10n.currentSlide ) , ( "exitFullscreen", E.string coursePresentationL10n.exitFullscreen ) , ( "exportAnswers", E.string coursePresentationL10n.exportAnswers ) , ( "fullscreen", E.string coursePresentationL10n.fullscreen ) , ( "hideKeywords", E.string coursePresentationL10n.hideKeywords ) , ( "lastSlide", E.string coursePresentationL10n.lastSlide ) , ( "maxScore", E.string coursePresentationL10n.maxScore ) , ( "nextSlide", E.string coursePresentationL10n.nextSlide ) , ( "noTitle", E.string coursePresentationL10n.noTitle ) , ( "prevSlide", E.string coursePresentationL10n.prevSlide ) , ( "printAllSlides", E.string coursePresentationL10n.printAllSlides ) , ( "printCurrentSlide", E.string coursePresentationL10n.printCurrentSlide ) , ( "printIngress", E.string coursePresentationL10n.printIngress ) , ( "printTitle", E.string coursePresentationL10n.printTitle ) , ( "retry", E.string coursePresentationL10n.retry ) , ( "score", E.string coursePresentationL10n.score ) , ( "scoreMessage", E.string coursePresentationL10n.scoreMessage ) , ( "shareFacebook", E.string coursePresentationL10n.shareFacebook ) , ( "shareGoogle", E.string coursePresentationL10n.shareGoogle ) , ( "shareResult", E.string coursePresentationL10n.shareResult ) , ( "shareTwitter", E.string coursePresentationL10n.shareTwitter ) , ( "showKeywords", E.string coursePresentationL10n.showKeywords ) , ( "showSolutions", E.string coursePresentationL10n.showSolutions ) , ( "slide", E.string coursePresentationL10n.slide ) , ( "slideCount", E.string coursePresentationL10n.slideCount ) , ( "solutionModeText", E.string coursePresentationL10n.solutionModeText ) , ( "solutionModeTitle", E.string coursePresentationL10n.solutionModeTitle ) , ( "solutionsButtonTitle", E.string coursePresentationL10n.solutionsButtonTitle ) , ( "summary", E.string coursePresentationL10n.summary ) , ( "summaryMultipleTaskText", E.string coursePresentationL10n.summaryMultipleTaskText ) , ( "total", E.string coursePresentationL10n.total ) , ( "totalScore", E.string coursePresentationL10n.totalScore ) , ( "yourScore", E.string coursePresentationL10n.yourScore ) ] encodedCoursePresentationOverride : CoursePresentationOverride -> E.Value encodedCoursePresentationOverride coursePresentationOverride = E.object [ ( "activeSurface", E.bool coursePresentationOverride.activeSurface ) , ( "enablePrintButton", E.bool coursePresentationOverride.enablePrintButton ) , ( "hideSummarySlide", E.bool coursePresentationOverride.hideSummarySlide ) , ( "social", encodedCoursePresentationOverrideSocial coursePresentationOverride.social ) , ( "summarySlideRetryButton", E.bool coursePresentationOverride.summarySlideRetryButton ) , ( "summarySlideSolutionButton", E.bool coursePresentationOverride.summarySlideSolutionButton ) ] encodedCoursePresentationOverrideSocial : CoursePresentationOverrideSocial -> E.Value encodedCoursePresentationOverrideSocial coursePresentationOverrideSocial = E.object [ ( "facebookShare", encodedCoursePresentationOverrideSocialFacebookShare coursePresentationOverrideSocial.facebookShare ) , ( "googleShareUrl", E.string coursePresentationOverrideSocial.googleShareUrl ) , ( "showFacebookShare", E.bool coursePresentationOverrideSocial.showFacebookShare ) , ( "showGoogleShare", E.bool coursePresentationOverrideSocial.showGoogleShare ) , ( "showTwitterShare", E.bool coursePresentationOverrideSocial.showTwitterShare ) , ( "twitterShare", encodedCoursePresentationOverrideSocialTwitterShare coursePresentationOverrideSocial.twitterShare ) ] encodedCoursePresentationOverrideSocialFacebookShare : CoursePresentationOverrideSocialFacebookShare -> E.Value encodedCoursePresentationOverrideSocialFacebookShare coursePresentationOverrideSocialFacebookShare = E.object [ ( "quote", E.string coursePresentationOverrideSocialFacebookShare.quote ) , ( "url", E.string coursePresentationOverrideSocialFacebookShare.url ) ] encodedCoursePresentationOverrideSocialTwitterShare : CoursePresentationOverrideSocialTwitterShare -> E.Value encodedCoursePresentationOverrideSocialTwitterShare coursePresentationOverrideSocialTwitterShare = E.object [ ( "hashtags", E.string coursePresentationOverrideSocialTwitterShare.hashtags ) , ( "statement", E.string coursePresentationOverrideSocialTwitterShare.statement ) , ( "url", E.string coursePresentationOverrideSocialTwitterShare.url ) ] encodedCoursePresentationPresentation : CoursePresentationPresentation -> E.Value encodedCoursePresentationPresentation coursePresentationPresentation = E.object [ ( "globalBackgroundSelector", encodedCoursePresentationPresentationGlobalBackgroundSelector coursePresentationPresentation.globalBackgroundSelector ) , ( "keywordListAlwaysShow", E.bool coursePresentationPresentation.keywordListAlwaysShow ) , ( "keywordListAutoHide", E.bool coursePresentationPresentation.keywordListAutoHide ) , ( "keywordListEnabled", E.bool coursePresentationPresentation.keywordListEnabled ) , ( "keywordListOpacity", E.int coursePresentationPresentation.keywordListOpacity ) , ( "slides", E.list encodedCoursePresentationPresentationSlidesObject coursePresentationPresentation.slides ) ] encodedCoursePresentationPresentationGlobalBackgroundSelector : CoursePresentationPresentationGlobalBackgroundSelector -> E.Value encodedCoursePresentationPresentationGlobalBackgroundSelector coursePresentationPresentationGlobalBackgroundSelector = E.object [ ( "fillGlobalBackground", E.string coursePresentationPresentationGlobalBackgroundSelector.fillGlobalBackground ) ] encodedCoursePresentationPresentationSlidesObject : CoursePresentationPresentationSlidesObject -> E.Value encodedCoursePresentationPresentationSlidesObject coursePresentationPresentationSlidesObject = E.object [ ( "elements", E.list (\_ -> E.null) coursePresentationPresentationSlidesObject.elements ) , ( "slideBackgroundSelector", encodedCoursePresentationPresentationSlidesObjectSlideBackgroundSelector coursePresentationPresentationSlidesObject.slideBackgroundSelector ) ] encodedCoursePresentationPresentationSlidesObjectSlideBackgroundSelector : CoursePresentationPresentationSlidesObjectSlideBackgroundSelector -> E.Value encodedCoursePresentationPresentationSlidesObjectSlideBackgroundSelector coursePresentationPresentationSlidesObjectSlideBackgroundSelector = E.object [ ( "fillSlideBackground", E.string coursePresentationPresentationSlidesObjectSlideBackgroundSelector.fillSlideBackground ) ] nouveauCoursePresentation = { l10n = { accessibilityCanvasLabel = "Le champs de présentation. Utilisez les fleches gauche et droite pour naviguer entre les diapositives." , accessibilityEnteredFullscreen = "Mode plein-écran activé" , accessibilityExitedFullscreen = "Mode plein-écran désactivé" , accessibilitySlideNavigationExplanation = "Utilisez les fleches gauche et droite pour pour naviguer entre les diapositives" , accessibilityTotalScore = "Vous avez obtenu @score sur @maxScore points au total" , confirmDialogConfirmText = "Envoyer et voir les résultats" , confirmDialogHeader = "Envoyer vos réponses" , confirmDialogText = "Cette action va envoyer vos réponses, voulez-vous continuer?" , containsCompleted = "@slideName contient des interactions complètes" , containsIncorrectAnswers = "@slideName contient des réponses incorrectes" , containsNotCompleted = "@slideName contient des interactions incomplètes" , containsOnlyCorrect = "toutes les réponses sont bonnes sur @slideName" , currentSlide = "Diapositive courante" , exitFullscreen = "Quitter le plein écran" , exportAnswers = "Exporter" , fullscreen = "Plein écran" , hideKeywords = "Cacher la liste des mots-clés" , lastSlide = "Dernière diapositive" , maxScore = "Score maximum" , nextSlide = "Diapositive suivante" , noTitle = "Sans intitulé" , prevSlide = "Diapositive précédente" , printAllSlides = "Imprimer toutes les diapositives" , printCurrentSlide = "Imprimer la diapositive courante" , printIngress = "Comment souhaitez-vous imprimer cette présentation ?" , printTitle = "Imprimer" , retry = "Recommencer" , score = "Score" , scoreMessage = "Votre score :" , shareFacebook = "Partager sur Facebook" , shareGoogle = "Partager sur Google+" , shareResult = "Partager le résultat" , shareTwitter = "Partager sur Twitter" , showKeywords = "Afficher la liste des mots-clés" , showSolutions = "Voir la correction" , slide = "Diapositive" , slideCount = "Diapositive a @index de @total" , solutionModeText = "Passer en mode "Correction"" , solutionModeTitle = "Sortir du mode "Correction"" , solutionsButtonTitle = "Afficher les commentaires" , summary = "Résumé" , summaryMultipleTaskText = "Activités multiples" , total = "Total" , totalScore = "Score total" , yourScore = "Votre score" } , override = { activeSurface = False , enablePrintButton = False , hideSummarySlide = False , social = { facebookShare = { quote = "I scored @score out of @maxScore on a task at @currentpageurl." , url = "@currentpageurl" } , googleShareUrl = "@currentpageurl" , showFacebookShare = False , showGoogleShare = False , showTwitterShare = False , twitterShare = { hashtags = "h5p, course" , statement = "I scored @score out of @maxScore on a task at @currentpageurl." , url = "@currentpageurl" } } , summarySlideRetryButton = True , summarySlideSolutionButton = True } , presentation = { globalBackgroundSelector = { fillGlobalBackground = "" } , keywordListAlwaysShow = False , keywordListAutoHide = False , keywordListEnabled = True , keywordListOpacity = 90 , slides = [ { elements = [] , slideBackgroundSelector = { fillSlideBackground = "" } } ] } } {- ████████ ██████ ██ ██ ███████ ██ ███████ █████ ██ ███████ ███████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██ ██ █████ ██ █████ ███████ ██ ███████ █████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ███████ ██ ██ ██ ██ ███████ ███████ ███████ ██████ ██ ██ ███████ ███████ ████████ ██ ██████ ███ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ████ ██ ██ ██ ██ ██ █████ ███████ ██ ██ ██ ██ ██ ██ ██ ██ ▄▄ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██████ ███████ ███████ ██ ██ ██████ ██ ████ ▀▀ -} type alias TrueFalse = { behaviour : TrueFalseBehaviour , confirmCheck : TrueFalseConfirmCheck , confirmRetry : TrueFalseConfirmRetry , correct : String , l10n : TrueFalseL10n , media : TrueFalseMedia , question : String } type alias TrueFalseBehaviour = { autoCheck : Bool , confirmCheckDialog : Bool , confirmRetryDialog : Bool , enableCheckButton : Bool , enableRetry : Bool , enableSolutionsButton : Bool , feedbackOnCorrect : Maybe String , feedbackOnWrong : Maybe String } type alias TrueFalseConfirmCheck = { body : String , cancelLabel : String , confirmLabel : String , header : String } type alias TrueFalseConfirmRetry = { body : String , cancelLabel : String , confirmLabel : String , header : String } type alias TrueFalseL10n = { a11yCheck : String , a11yRetry : String , a11yShowSolution : String , checkAnswer : String , correctAnswerMessage : String , falseText : String , score : String , scoreBarLabel : String , showSolutionButton : String , submitAnswer : String , trueText : String , tryAgain : String , wrongAnswerMessage : String } type alias TrueFalseMedia = { disableImageZooming : Bool } trueFalseDecoder : D.Decoder TrueFalse trueFalseDecoder = D.map7 TrueFalse (D.field "behaviour" trueFalseBehaviourDecoder) (D.field "confirmCheck" trueFalseConfirmCheckDecoder) (D.field "confirmRetry" trueFalseConfirmRetryDecoder) (D.field "correct" D.string) (D.field "l10n" trueFalseL10nDecoder) (D.field "media" trueFalseMediaDecoder) (D.field "question" D.string) trueFalseBehaviourDecoder : D.Decoder TrueFalseBehaviour trueFalseBehaviourDecoder = D.map8 TrueFalseBehaviour (D.field "autoCheck" D.bool) (D.field "confirmCheckDialog" D.bool) (D.field "confirmRetryDialog" D.bool) (D.field "enableCheckButton" D.bool) (D.field "enableRetry" D.bool) (D.field "enableSolutionsButton" D.bool) (D.maybe <| D.field "feedbackOnCorrect" D.string) (D.maybe <| D.field "feedbackOnWrong" D.string) trueFalseConfirmCheckDecoder : D.Decoder TrueFalseConfirmCheck trueFalseConfirmCheckDecoder = D.map4 TrueFalseConfirmCheck (D.field "body" D.string) (D.field "cancelLabel" D.string) (D.field "confirmLabel" D.string) (D.field "header" D.string) trueFalseConfirmRetryDecoder : D.Decoder TrueFalseConfirmRetry trueFalseConfirmRetryDecoder = D.map4 TrueFalseConfirmRetry (D.field "body" D.string) (D.field "cancelLabel" D.string) (D.field "confirmLabel" D.string) (D.field "header" D.string) trueFalseL10nDecoder : D.Decoder TrueFalseL10n trueFalseL10nDecoder = let fieldSet0 = D.map8 TrueFalseL10n (D.field "a11yCheck" D.string) (D.field "a11yRetry" D.string) (D.field "a11yShowSolution" D.string) (D.field "checkAnswer" D.string) (D.field "correctAnswerMessage" D.string) (D.field "falseText" D.string) (D.field "score" D.string) (D.field "scoreBarLabel" D.string) in D.map6 (<|) fieldSet0 (D.field "showSolutionButton" D.string) (D.field "submitAnswer" D.string) (D.field "trueText" D.string) (D.field "tryAgain" D.string) (D.field "wrongAnswerMessage" D.string) trueFalseMediaDecoder : D.Decoder TrueFalseMedia trueFalseMediaDecoder = D.map TrueFalseMedia (D.field "disableImageZooming" D.bool) encodedTrueFalse : TrueFalse -> E.Value encodedTrueFalse trueFalse = E.object [ ( "behaviour", encodedTrueFalseBehaviour trueFalse.behaviour ) , ( "confirmCheck", encodedTrueFalseConfirmCheck trueFalse.confirmCheck ) , ( "confirmRetry", encodedTrueFalseConfirmRetry trueFalse.confirmRetry ) , ( "correct", E.string trueFalse.correct ) , ( "l10n", encodedTrueFalseL10n trueFalse.l10n ) , ( "media", encodedTrueFalseMedia trueFalse.media ) , ( "question", E.string trueFalse.question ) ] encodedTrueFalseBehaviour : TrueFalseBehaviour -> E.Value encodedTrueFalseBehaviour trueFalseBehaviour = E.object <| [ ( "autoCheck", E.bool trueFalseBehaviour.autoCheck ) , ( "confirmCheckDialog", E.bool trueFalseBehaviour.confirmCheckDialog ) , ( "confirmRetryDialog", E.bool trueFalseBehaviour.confirmRetryDialog ) , ( "enableCheckButton", E.bool trueFalseBehaviour.enableCheckButton ) , ( "enableRetry", E.bool trueFalseBehaviour.enableRetry ) , ( "enableSolutionsButton", E.bool trueFalseBehaviour.enableSolutionsButton ) ] ++ (case trueFalseBehaviour.feedbackOnCorrect of Just value -> [ ( "feedbackOnCorrect", E.string value ) ] Nothing -> [] ) ++ (case trueFalseBehaviour.feedbackOnWrong of Just value -> [ ( "feedbackOnWrong", E.string value ) ] Nothing -> [] ) encodedTrueFalseConfirmCheck : TrueFalseConfirmCheck -> E.Value encodedTrueFalseConfirmCheck trueFalseConfirmCheck = E.object [ ( "body", E.string trueFalseConfirmCheck.body ) , ( "cancelLabel", E.string trueFalseConfirmCheck.cancelLabel ) , ( "confirmLabel", E.string trueFalseConfirmCheck.confirmLabel ) , ( "header", E.string trueFalseConfirmCheck.header ) ] encodedTrueFalseConfirmRetry : TrueFalseConfirmRetry -> E.Value encodedTrueFalseConfirmRetry trueFalseConfirmRetry = E.object [ ( "body", E.string trueFalseConfirmRetry.body ) , ( "cancelLabel", E.string trueFalseConfirmRetry.cancelLabel ) , ( "confirmLabel", E.string trueFalseConfirmRetry.confirmLabel ) , ( "header", E.string trueFalseConfirmRetry.header ) ] encodedTrueFalseL10n : TrueFalseL10n -> E.Value encodedTrueFalseL10n trueFalseL10n = E.object [ ( "a11yCheck", E.string trueFalseL10n.a11yCheck ) , ( "a11yRetry", E.string trueFalseL10n.a11yRetry ) , ( "a11yShowSolution", E.string trueFalseL10n.a11yShowSolution ) , ( "checkAnswer", E.string trueFalseL10n.checkAnswer ) , ( "correctAnswerMessage", E.string trueFalseL10n.correctAnswerMessage ) , ( "falseText", E.string trueFalseL10n.falseText ) , ( "score", E.string trueFalseL10n.score ) , ( "scoreBarLabel", E.string trueFalseL10n.scoreBarLabel ) , ( "showSolutionButton", E.string trueFalseL10n.showSolutionButton ) , ( "submitAnswer", E.string trueFalseL10n.submitAnswer ) , ( "trueText", E.string trueFalseL10n.trueText ) , ( "tryAgain", E.string trueFalseL10n.tryAgain ) , ( "wrongAnswerMessage", E.string trueFalseL10n.wrongAnswerMessage ) ] encodedTrueFalseMedia : TrueFalseMedia -> E.Value encodedTrueFalseMedia trueFalseMedia = E.object [ ( "disableImageZooming", E.bool trueFalseMedia.disableImageZooming ) ] nouveauTrueFalse = { behaviour = { autoCheck = True , confirmCheckDialog = False , confirmRetryDialog = False , enableCheckButton = True , enableRetry = True , enableSolutionsButton = True , feedbackOnCorrect = Just "C'est la base !\n" , feedbackOnWrong = Nothing } , confirmCheck = { body = "Êtes-vous sûr de vouloir terminer ?" , cancelLabel = "Annuler" , confirmLabel = "Confirmer" , header = "Terminer ?" } , confirmRetry = { body = "Êtes-vous sûr de vouloir recommencer ?" , cancelLabel = "Annuler" , confirmLabel = "Confirmer" , header = "Recommencer ?" } , correct = "true" , l10n = { a11yCheck = "Check the answers. The responses will be marked as correct, incorrect, or unanswered." , a11yRetry = "Retry the task. Reset all responses and start the task over again." , a11yShowSolution = "Show the solution. The task will be marked with its correct solution." , checkAnswer = "Vérifier" , correctAnswerMessage = "Bonne réponse" , falseText = "Faux" , score = "Vous avez obtenu @score points sur un total de @total" , scoreBarLabel = "Vous avez obtenu @score points sur un total de @total" , showSolutionButton = "Voir la solution" , submitAnswer = "Vérifier" , trueText = "Vrai" , tryAgain = "Recommencer" , wrongAnswerMessage = "Réponse incorrecte" } , media = { disableImageZooming = False } , question = "" } {- ██████╗ ██╗ ██╗██╗██╗ ██████╗ ███████╗██████╗ ███████╗ ██╔══██╗██║ ██║██║██║ ██╔══██╗██╔════╝██╔══██╗██╔════╝ ██████╔╝██║ ██║██║██║ ██║ ██║█████╗ ██████╔╝███████╗ ██╔══██╗██║ ██║██║██║ ██║ ██║██╔══╝ ██╔══██╗╚════██║ ██████╔╝╚██████╔╝██║███████╗██████╔╝███████╗██║ ██║███████║ ╚═════╝ ╚═════╝ ╚═╝╚══════╝╚═════╝ ╚══════╝╚═╝ ╚═╝╚══════╝ -} type H5pTree = H5pTree Context String (A.Array H5pTree) fromH5pTree tree = case tree of H5pTree BranchingScenarioContext title subTrees -> BranchingScenarioH5P (nouveauBranchingScenario |> withMap startScreenField startScreenSubtitleField title |> .with contentField (A.map fromBranchingScenario subTrees) ) H5pTree CoursePresentationContext title subTrees -> CoursePresentationH5P nouveauCoursePresentation H5pTree TrueFalseContext title subTrees -> TrueFalseH5P nouveauTrueFalse _ -> TrueFalseH5P nouveauTrueFalse {- { nouveauBranchingScenario | startScreen = nouveauBranchingScenario.startScreen |> withStartScreenSubtitle title , content = L.map fromH5Ptree subTrees } -} {- H5Ptree CoursePresentationContext title subTrees -> CoursePresentationH5P nouveauCoursePresentation H5Ptree TrueFalseContext title subTrees -> TrueFalseH5P nouveauTrueFalse _ -> TrueFalseH5P nouveauTrueFalse -} fromBranchingScenario subTree = case subTree of H5pTree CoursePresentationContext title subTrees -> { contentBehaviour = "useBehavioural" , feedback = { subtitle = "" } , forceContentFinished = "useBehavioural" , showContentTitle = False , type_ = { library = "H5P.CoursePresentation 1.24" , metadata = { contentType = "Branching Question" , license = "U" , title = "Untitled Branching Question" } , params = CoursePresentationH5P nouveauCoursePresentation , subContentId = uuid 1 } } H5pTree context title subTrees -> { contentBehaviour = "useBehavioural" , feedback = { subtitle = "" } , forceContentFinished = "useBehavioural" , showContentTitle = False , type_ = { library = "H5P.CoursePresentation 1.24" , metadata = { contentType = "Branching Question" , license = "U" , title = "Untitled Branching Question" } , params = CoursePresentationH5P nouveauCoursePresentation , subContentId = uuid 1 } } withMap field fieldInside value record = field.with (fieldInside.with value (field.field record)) record startScreenField = { with = \value record -> { record | startScreen = value } , field = .startScreen } startScreenSubtitleField = { with = \value record -> { record | startScreenSubtitle = value } , field = .startScreenSubtitle } contentField = { with = \value record -> { record | content = value } , field = .content } {- .--..--..--..--..--..--..--..--..--..--..--..--..--..--..--..--. / .. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \.. \ \ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/ / \/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ / / /\/ /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /`' /\/ /\ / /\ \/`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'\ \/\ \ \ \/\ \ /\ \/ / \/ /\ \ ██████╗ █████╗ ██████╗ ███████╗███████╗██████╗ / /\/ / / /\/ / ██╔══██╗██╔══██╗██╔══██╗██╔════╝██╔════╝██╔══██╗ \ \/ /\ / /\ \/ ██████╔╝███████║██████╔╝███████╗█████╗ ██████╔╝ \ \/\ \ \ \/\ \ ██╔═══╝ ██╔══██║██╔══██╗╚════██║██╔══╝ ██╔══██╗ /\ \/ / \/ /\ \ ██║ ██║ ██║██║ ██║███████║███████╗██║ ██║ / /\/ / / /\/ / ╚═╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚══════╝╚══════╝╚═╝ ╚═╝ \ \/ /\ / /\ \/ \ \/\ \ \ \/\ \.--..--..--..--..--..--..--..--..--..--..--..--..--./\ \/ / \/ /\/ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ ../ /\/ / / /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\/ /\ / /\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \/\ \ \ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `'\ `' / `--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--'`--' -} type Context = PreambleContext | RootContext | BranchingScenarioContext | BranchingQuestionContext | BranchingQuestionAlternativeContext | CoursePresentationContext | TrueFalseContext parser = succeed (L.map fromH5pTree) |. preambleParser |= loop (State 0 0 []) contentsParser |. end EndOfFile type alias State = { depth : Int , maxDepth : Int , h5pTree : List H5pTree } preambleParser = inContext PreambleContext <| -- Plus compliqué que nécessaire, en vue d'amélioration succeed identity |. whileNoStarOnFirstColumnOrEndOfFile contentsParser profondeur context state = succeed Tuple.pair |= countStars |= contentParser |> andThen (\( numberOfStars, content ) -> if numberOfStars <= 0 then state.h5pTree else if Dict.member numberOfStars State.starsDepth then state.h5pTree |> A.set A.push content state.h5pTree |> Done |> succeed else if numberOfStars < L.maximum (Dict.keys state.starsDepth) then problem InconsistantStructure else succeed (\content -> Loop <| State (content :: state.contents) numberOfStars) ) contentParser profondeur context = succeed identity |. espaces |= getChompedString (chompWhile (\c -> c /= ' ' && c /= '\n' && c /= '\u{000D}' && c /= '\u{000D}' ) ) |> andThen (\maybeContentType -> case ( maybeContentType, context ) of ( "BranchingScenario", RootContext ) -> contentParserHelp profondeur BranchingScenarioContext "" ( "BranchingScenario", _ ) -> problem (Problem "Un BranchingScenario doit se trouver à la racine") ( "CoursePresentation", RootContext ) -> contentParserHelp profondeur CoursePresentationContext "" ( "CoursePresentation", BranchingScenarioContext ) -> contentParserHelp profondeur CoursePresentationContext "" ( "CoursePresentation", BranchingQuestionAlternativeContext ) -> contentParserHelp profondeur CoursePresentationContext "" ( "CoursePresentation", _ ) -> problem (Problem "Un CoursePresentation doit se trouver à la racine, sous un BranchingScenario ou dans une alternative de BranchingQuestion") ( "TrueFalse", RootContext ) -> contentParserHelp profondeur TrueFalseContext "" ( "TrueFalse", CoursePresentationContext ) -> contentParserHelp profondeur TrueFalseContext "" ( "TrueFalse", _ ) -> problem (Problem "Un TrueFalse doit se trouver à la racine ou dans un CoursePresentation") -- Dans les trois cas ci-dessous, le maybeContentType ne désigne -- pas un contentType, c'est une astuce pour récupérer le texte avaler. -- (cf. bit dans la définition de contentParserHelp) ( _, BranchingScenarioContext ) -> contentParserHelp profondeur BranchingQuestionContext maybeContentType ( _, BranchingQuestionAlternativeContext ) -> contentParserHelp profondeur BranchingQuestionContext maybeContentType ( _, BranchingQuestionContext ) -> contentParserHelp profondeur BranchingQuestionAlternativeContext maybeContentType _ -> problem <| UnknownContentType maybeContentType ) test = """* BranchingScenario Titre ** BranchingQuestion Question *** trueFalse **** CoursePresentation ***** TrueFalse * CoursePresentation""" contentParserHelp profondeurAdegager context bit = --TODO let f endOfLine contentList = H5pTree context (bit ++ endOfLine) contentList in inContext context <| succeed f |= tillEndOfLine |. whileNoStarOnFirstColumnOrEndOfFile whileNoStarOnFirstColumn = succeed identity |. chompWhile ((/=) '*') |= getCol |> andThen (\col -> if col > 1 then succeed () |. symbol (Token "*" EndOfFile) |. whileNoStarOnFirstColumn else succeed () ) whileNoStarOnFirstColumnOrEndOfFile = succeed identity |. chompWhile ((/=) '*') |= getCol |> andThen (\col -> if col > 1 then oneOf [ end EndOfFile , succeed () |. token (Token "*" EndOfFile) |. whileNoStarOnFirstColumnOrEndOfFile ] else succeed () ) countStars = succeed S.length |= getChompedString (chompWhile ((==) '*')) stars profondeur = succeed () |. symbol (Token (S.repeat profondeur "*") GenericProblem) |. espaces tillEndOfLine = getChompedString <| succeed () |. chompWhile ((/=) '\n') {-| Avale tout l'espace blanc et impose de s'arrêter soit en début de ligne soit en fin de fichier. -} blankLines = succeed () |. chompWhile (\x -> x == ' ' || x == '\t' || x == '\n' || x == '\u{000D}' ) |. oneOf [ end GenericProblem , getPosition |> andThen (\( row, col ) -> if col == 1 then succeed () else problemBis ("N'y aurait-il pas des espaces en trop au début de la ligne " ++ String.fromInt row ++ " ?" ) ) ] espaces = chompWhile <| \x -> x == ' ' || x == '\t' keywordBis x = keyword <| Token x GenericProblem problemBis x = problem <| Problem x {- ██████ ██████ █████ ███ ██ ██████ ██ ██ ██ ███ ██ ██████ ██ ██ ██ ██ ██ ██ ████ ██ ██ ██ ██ ██ ████ ██ ██ ██████ ██████ ███████ ██ ██ ██ ██ ███████ ██ ██ ██ ██ ██ ███ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██████ ██ ██ ██ ██ ██ ████ ██████ ██ ██ ██ ██ ████ ██████ ███████ ██████ ███████ ███ ██ █████ ██████ ██ ██████ ██ ██ ██ ████ ██ ██ ██ ██ ██ ██ ██ ██ ███████ ██ █████ ██ ██ ██ ███████ ██████ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ███████ ██████ ███████ ██ ████ ██ ██ ██ ██ ██ ██████ -} {- branchingScenarioParser profondeur = let f title contentList = BranchingScenarioH5P { nouveauBranchingScenario | startScreen = nouveauBranchingScenario.startScreen |> withStartScreenSubtitle title , content = [] -- TODO } in succeed f |= titleParser |. blankLines |= loop [] (contentsParser (profondeur + 1) BranchingScenarioContext) -} --|= loop ( [], 0 ) (branchingScenarioParserHelp (profondeur + 1)) --|. blankLines {- Dans une configuration de la forme : *BranchingScenario Titre du cours ** BranchingQuestion Cxu ? ... ** CoursePresentation ... Récupère tout ce qui se trouve au niveau ** sous la forme d'une liste -} {- branchingScenarioParserHelp profondeur ( contentList, contentId ) = oneOf [ let f ( contents, id ) = Loop ( contentList ++ contents, id + 1 ) in succeed f |= contentParser profondeur contentId |. blankLines , succeed (Done contentList) ] -} {- contentParserBis profondeur contentId = oneOf [ backtrackable <| branchingQuestionParser profondeur contentId , succeed (\x -> ( [ x ], contentId + 1 )) |= oneOf [ backtrackable <| coursePresentationParser profondeur , trueFalseParser profondeur ] --, problemBis "Oups" ] -} uuid n = Random.initialSeed n |> Random.step UUID.generator |> Tuple.first |> UUID.toString {- ██████╗ ██████╗ █████╗ ███╗ ██╗ ██████╗██╗ ██╗██╗███╗ ██╗ ██████╗ ██╔══██╗██╔══██╗██╔══██╗████╗ ██║██╔════╝██║ ██║██║████╗ ██║██╔════╝ ██████╔╝██████╔╝███████║██╔██╗ ██║██║ ███████║██║██╔██╗ ██║██║ ███╗ ██╔══██╗██╔══██╗██╔══██║██║╚██╗██║██║ ██╔══██║██║██║╚██╗██║██║ ██║ ██████╔╝██║ ██║██║ ██║██║ ╚████║╚██████╗██║ ██║██║██║ ╚████║╚██████╔╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚═╝ ╚═══╝ ╚═════╝╚═╝ ╚═╝╚═╝╚═╝ ╚═══╝ ╚═════╝ ██████╗ ██╗ ██╗███████╗███████╗████████╗██╗ ██████╗ ███╗ ██╗ ██╔═══██╗██║ ██║██╔════╝██╔════╝╚══██╔══╝██║██╔═══██╗████╗ ██║ ██║ ██║██║ ██║█████╗ ███████╗ ██║ ██║██║ ██║██╔██╗ ██║ ██║▄▄ ██║██║ ██║██╔══╝ ╚════██║ ██║ ██║██║ ██║██║╚██╗██║ ╚██████╔╝╚██████╔╝███████╗███████║ ██║ ██║╚██████╔╝██║ ╚████║ ╚══▀▀═╝ ╚═════╝ ╚══════╝╚══════╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═══╝ -} {- {-| Ici branchList est un enregistrement forçant la liste à avoir au moins deux éléments -} branchingQuestionParser profondeur contentId = let f question ( branchList, id ) = ( branchList.first :: branchList.second :: branchList.others , contentId + L.length branchList.others + 2 ) in succeed f |. stars profondeur |. keywordBis "BranchingQuestion" |. espaces |= questionParser |. blankLines -- Je dirais qu'il faut un contentId + 1 ici |= loop ( [], contentId ) (branchingQuestionAlternativeParser (profondeur + 1)) {-| Ici branchList est une liste -} branchingQuestionAlternativeParser profondeur ( branchList, contentId ) = oneOf [ let f alternative ( alternativeList, id ) = case branchList of [] -> -- TODO Loop ( alternativeList, id ) _ -> -- TODO Loop ( alternativeList, id ) in succeed f |. stars profondeur |= alternativeAnswerParser |= loop ( [], contentId ) (branchingQuestionAlternativeParserHelp (profondeur + 1)) , (succeed () |> P.map (\_ -> List.concat branchList) ) |> andThen (\xx -> case xx of [] -> problemBis "Un embranchement doit avoir des branches !" x :: [] -> problemBis "Un embranchement doit avoir au moins deux branches !" x :: y :: zz -> succeed () |> P.map (\_ -> Done ( { first = x, second = y, others = zz }, contentId )) ) ] {-| Tout le travail reste à faire ! -} branchingQuestionAlternativeParserHelp profondeur ( branchList, contentId ) = oneOf [ succeed (Done ( [ [ CoursePresentationH5P nouveauCoursePresentation ] , [ TrueFalseH5P nouveauTrueFalse ] ] , contentId ) ) ] -} {- ██████╗ ███████╗███████╗████████╗██╗ ██████╗ ███╗ ██╗ ██████╗ ███████╗███████╗ ██╔════╝ ██╔════╝██╔════╝╚══██╔══╝██║██╔═══██╗████╗ ██║ ██╔══██╗██╔════╝██╔════╝ ██║ ███╗█████╗ ███████╗ ██║ ██║██║ ██║██╔██╗ ██║ ██║ ██║█████╗ ███████╗ ██║ ██║██╔══╝ ╚════██║ ██║ ██║██║ ██║██║╚██╗██║ ██║ ██║██╔══╝ ╚════██║ ╚██████╔╝███████╗███████║ ██║ ██║╚██████╔╝██║ ╚████║ ██████╔╝███████╗███████║ ╚═════╝ ╚══════╝╚══════╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═══╝ ╚═════╝ ╚══════╝╚══════╝ ███████╗██████╗ ██████╗ ███████╗██╗ ██╗██████╗ ███████╗ ██╔════╝██╔══██╗██╔══██╗██╔════╝██║ ██║██╔══██╗██╔════╝ █████╗ ██████╔╝██████╔╝█████╗ ██║ ██║██████╔╝███████╗ ██╔══╝ ██╔══██╗██╔══██╗██╔══╝ ██║ ██║██╔══██╗╚════██║ ███████╗██║ ██║██║ ██║███████╗╚██████╔╝██║ ██║███████║ ╚══════╝╚═╝ ╚═╝╚═╝ ╚═╝╚══════╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ -} type Problem = NoContent | BadKeyword String | Problem String | GenericProblem | EndOfFile | ExpectingContentType | UnknownContentType String | InconsistantStructure deadEndsToStringBis errs = errs |> L.map voirErreur |> S.join "\n\n" |> (++) "J'ai rencontré les problèmes suivants :\n\n" voirErreur err = "Problème : " ++ showProblem err.problem ++ "Ligne : " ++ String.fromInt err.row ++ " | Colonne : " ++ String.fromInt err.col ++ showContext (L.map .context err.contextStack) ++ "\n\n-----------------------------------------\n" showProblem prob = case prob of Problem p -> p ++ "\n" NoContent -> """Je n'ai aucun contenu à produire ! Y a-t-il autre chose qu'un préambule à analyser ? Est-ce que vos * ne seraient pas trop indentées ? """ EndOfFile -> "Fin de fichier\n" ExpectingContentType -> """Je m'attends à trouver l'un des mots clefs suivants : BranchingScenario CoursePresentation TrueFalse """ UnknownContentType x -> "Contenu H5P inconnu : " ++ x ++ "\n" InconsistantStructure -> "La structure du document n'est pas consistante !" _ -> "Problème inconnu\n" showContext ccc = case ccc of [] -> "" _ -> "\nContexte :\n" ++ showContextHelp (L.reverse ccc) showContextHelp ccc = case ccc of [] -> "\n" PreambleContext :: cc -> "Préambule" RootContext :: cc -> showContextHelp cc BranchingScenarioContext :: cc -> "BranchingScenario > " ++ showContextHelp cc BranchingQuestionContext :: cc -> "BranchingQuestion > " ++ showContextHelp cc BranchingQuestionAlternativeContext :: cc -> "Alternative > " ++ showContextHelp cc CoursePresentationContext :: cc -> "CoursePresentation > " ++ showContextHelp cc TrueFalseContext :: cc -> "TrueFalse"