/* eslint-disable jsx-a11y/no-static-element-interactions */ /* eslint-disable jsx-a11y/click-events-have-key-events */ /* eslint-disable react/jsx-props-no-spreading */ import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import { computed, makeObservable } from 'mobx'; import { observer } from 'mobx-react'; import * as React from 'react'; import { emptyFunction } from '../../../Utils'; import { StopEvent, returnFalse, returnOne, returnZero } from '../../../ClientUtils'; import { Doc, Opt } from '../../../fields/Doc'; import { DocCast, NumCast, ScriptCast, StrCast } from '../../../fields/Types'; import { DocumentType } from '../../documents/DocumentTypes'; import { DragManager } from '../../util/DragManager'; import { StyleProp } from '../StyleProp'; import { DocumentView } from '../nodes/DocumentView'; import { FieldViewProps } from '../nodes/FieldView'; import { FormattedTextBox } from '../nodes/formattedText/FormattedTextBox'; import './CollectionCarouselView.scss'; import { CollectionSubView } from './CollectionSubView'; @observer export class CollectionCarouselView extends CollectionSubView() { private _dropDisposer?: DragManager.DragDropDisposer; constructor(props: any) { super(props); makeObservable(this); } componentWillUnmount() { this._dropDisposer?.(); } protected createDashEventsTarget = (ele: HTMLDivElement | null) => { this._dropDisposer?.(); if (ele) { this._dropDisposer = DragManager.MakeDropTarget(ele, this.onInternalDrop.bind(this), this.layoutDoc); } }; @computed get carouselItems() { return this.childLayoutPairs.filter(pair => pair.layout.type !== DocumentType.LINK); } /** * Goes to the next flashcard in the stack and filters * based on the the currently selected option. */ advance = (e: React.MouseEvent) => { e.stopPropagation(); this.layoutDoc._carousel_index = (NumCast(this.layoutDoc._carousel_index) + 1) % this.carouselItems.length; let startInd = this.layoutDoc._carousel_index; // if the star filter is selected if (this.layoutDoc.filterOp === 'star') { // go to a flashcard that is starred, skip the ones that aren't while (!this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_star`] && (startInd + 1) % this.carouselItems.length !== this.layoutDoc._carousel_index) { startInd = (startInd + 1) % this.carouselItems.length; } this.layoutDoc._carousel_index = startInd; // if there aren't any starred, show all cards if (!this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_star`]) { this.layoutDoc.filterOp = 'all'; } } // if the practice filter is selected if (this.layoutDoc.filterOp === 'practice') { // go to a new index that is missed, skip the ones that are correct while (this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_missed`] === 'correct' && (startInd + 1) % this.carouselItems.length !== this.layoutDoc._carousel_index) { startInd = (startInd + 1) % this.carouselItems.length; } this.layoutDoc._carousel_index = startInd; // if the user has gone through all of the cards and gotten them all correct, show all cards and exit practice mode if (this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_missed`] === 'correct') { this.layoutDoc.filterOp = 'all'; // set all the cards to missed for (let i = 0; i < this.carouselItems.length; i++) { const curDoc = this.carouselItems?.[NumCast(i)]; curDoc.layout[`${this.fieldKey}_missed`] = undefined; } } } }; /** * Goes to the previous flashcard in the stack and filters * based on the the currently selected option. */ goback = (e: React.MouseEvent) => { e.stopPropagation(); this.layoutDoc._carousel_index = (NumCast(this.layoutDoc._carousel_index) - 1 + this.carouselItems.length) % this.carouselItems.length; let startInd = this.layoutDoc._carousel_index; // if the star filter is selected if (this.layoutDoc.filterOp === 'star') { // go to a new index that is starred, skip the ones that aren't while (!this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_star`] && (startInd - 1 + this.carouselItems.length) % this.carouselItems.length !== this.layoutDoc._carousel_index) { startInd = (startInd - 1 + this.carouselItems.length) % this.carouselItems.length; } this.layoutDoc._carousel_index = startInd; // if there aren't any starred, show all cards if (!this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_star`]) { this.layoutDoc.filterOp = 'all'; } } // if the practice filter is selected if (this.layoutDoc.filterOp === 'practice') { // go to a new index that is missed, skip the ones that are correct while (this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_missed`] === 'correct' && (startInd - 1 + this.carouselItems.length) % this.carouselItems.length !== this.layoutDoc._carousel_index) { startInd = (startInd - 1 + this.carouselItems.length) % this.carouselItems.length; } this.layoutDoc._carousel_index = startInd; // See all flashcards when finish going through practice mode and set all of the flashcards back to if (this.carouselItems?.[NumCast(startInd)].layout[`${this.fieldKey}_missed`] === 'correct') { this.layoutDoc.filterOp = 'all'; for (let i = 0; i < this.carouselItems.length; i++) { const curDoc = this.carouselItems?.[NumCast(i)]; curDoc.layout[`${this.fieldKey}_missed`] = undefined; } } } }; /* * Stars the document when the star button is pressed. */ star = (e: React.MouseEvent) => { e.stopPropagation(); const curDoc = this.carouselItems?.[NumCast(this.layoutDoc._carousel_index)]; if (!curDoc) return; if (curDoc.layout[`${this.fieldKey}_star`] === undefined) curDoc.layout[`${this.fieldKey}_star`] = true; else curDoc.layout[`${this.fieldKey}_star`] = !curDoc.layout[`${this.fieldKey}_star`]; }; /* * Sets a flashcard to either missed or correct depending on if they got the question right in practice mode. */ missed = (e: React.MouseEvent, val: string) => { e.stopPropagation(); const curDoc = this.carouselItems?.[NumCast(this.layoutDoc._carousel_index)]; curDoc.layout[`${this.fieldKey}_missed`] = val; this.advance(e); }; captionStyleProvider = (doc: Doc | undefined, captionProps: Opt, property: string): any => { // first look for properties on the document in the carousel, then fallback to properties on the container const childValue = doc?.['caption_' + property] ? this._props.styleProvider?.(doc, captionProps, property) : undefined; return childValue ?? this._props.styleProvider?.(this.layoutDoc, captionProps, property); }; panelHeight = () => this._props.PanelHeight() - (StrCast(this.layoutDoc._layout_showCaption) ? 50 : 0); onContentDoubleClick = () => ScriptCast(this.layoutDoc.onChildDoubleClick); onContentClick = () => ScriptCast(this.layoutDoc.onChildClick); @computed get marginX() { return NumCast(this.layoutDoc.caption_xMargin, 50); } captionWidth = () => this._props.PanelWidth() - 2 * this.marginX; @computed get content() { const index = NumCast(this.layoutDoc._carousel_index); const curDoc = this.carouselItems?.[index]; const captionProps = { ...this._props, NativeScaling: returnOne, PanelWidth: this.captionWidth, fieldKey: 'caption', setHeight: undefined, setContentView: undefined }; const carouselShowsCaptions = StrCast(this.layoutDoc._layout_showCaption); return !(curDoc?.layout instanceof Doc) ? null : ( <>
{!carouselShowsCaptions ? null : (
)} ); } @computed get buttons() { if (!this.carouselItems?.[NumCast(this.layoutDoc._carousel_index)]) return null; return ( <>
this.missed(e, 'missed')} style={{ visibility: this.layoutDoc.filterOp === 'practice' ? 'visible' : 'hidden' }}>
this.missed(e, 'correct')} style={{ visibility: this.layoutDoc.filterOp === 'practice' ? 'visible' : 'hidden' }}>
); } render() { return (
{this.content} {/* Displays a message to the user to add more flashcards if they are in practice mode and no flashcards are there. */}

Add flashcards!

{/* Displays a message to the user that a flashcard was recently missed if they had previously gotten it wrong. */}

Recently missed!

{this.Document._chromeHidden ? null : this.buttons}
); } }