2022-09-08 21:40:48 +02:00
|
|
|
import { Store, UIEventSource } from "../../Logic/UIEventSource"
|
|
|
|
import TagRenderingQuestion from "./TagRenderingQuestion"
|
|
|
|
import Translations from "../i18n/Translations"
|
|
|
|
import Combine from "../Base/Combine"
|
|
|
|
import BaseUIElement from "../BaseUIElement"
|
|
|
|
import { VariableUiElement } from "../Base/VariableUIElement"
|
|
|
|
import TagRenderingConfig from "../../Models/ThemeConfig/TagRenderingConfig"
|
|
|
|
import { Unit } from "../../Models/Unit"
|
|
|
|
import Lazy from "../Base/Lazy"
|
2023-01-06 03:37:22 +01:00
|
|
|
import { OsmServiceState } from "../../Logic/Osm/OsmConnection"
|
2020-10-27 01:01:34 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates all the questions, one by one
|
|
|
|
*/
|
2021-06-28 00:45:49 +02:00
|
|
|
export default class QuestionBox extends VariableUiElement {
|
2022-09-08 21:40:48 +02:00
|
|
|
public readonly skippedQuestions: UIEventSource<number[]>
|
|
|
|
public readonly restingQuestions: Store<BaseUIElement[]>
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
state,
|
|
|
|
options: {
|
|
|
|
tagsSource: UIEventSource<any>
|
|
|
|
tagRenderings: TagRenderingConfig[]
|
|
|
|
units: Unit[]
|
2023-03-08 01:36:27 +01:00
|
|
|
showAllQuestionsAtOnce?: boolean | Store<boolean>
|
2022-09-08 21:40:48 +02:00
|
|
|
}
|
|
|
|
) {
|
2021-06-28 00:45:49 +02:00
|
|
|
const skippedQuestions: UIEventSource<number[]> = new UIEventSource<number[]>([])
|
2020-10-27 01:01:34 +01:00
|
|
|
|
2021-11-10 18:42:31 +01:00
|
|
|
const tagsSource = options.tagsSource
|
|
|
|
const units = options.units
|
2021-11-12 01:44:13 +01:00
|
|
|
options.showAllQuestionsAtOnce = options.showAllQuestionsAtOnce ?? false
|
2021-11-10 18:42:31 +01:00
|
|
|
const tagRenderings = options.tagRenderings
|
2022-09-08 21:40:48 +02:00
|
|
|
.filter((tr) => tr.question !== undefined)
|
|
|
|
.filter((tr) => tr.question !== null)
|
|
|
|
|
2022-12-16 13:45:07 +01:00
|
|
|
let focus: () => void = () => {}
|
2022-12-06 03:43:54 +01:00
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
const tagRenderingQuestions = tagRenderings.map(
|
|
|
|
(tagRendering, i) =>
|
|
|
|
new Lazy(
|
|
|
|
() =>
|
|
|
|
new TagRenderingQuestion(tagsSource, tagRendering, state, {
|
|
|
|
units: units,
|
|
|
|
afterSave: () => {
|
|
|
|
// We save and indicate progress by pinging and recalculating
|
|
|
|
skippedQuestions.ping()
|
2022-12-06 03:43:54 +01:00
|
|
|
focus()
|
2022-09-08 21:40:48 +02:00
|
|
|
},
|
|
|
|
cancelButton: Translations.t.general.skip
|
|
|
|
.Clone()
|
|
|
|
.SetClass("btn btn-secondary")
|
|
|
|
.onClick(() => {
|
|
|
|
skippedQuestions.data.push(i)
|
|
|
|
skippedQuestions.ping()
|
2022-12-06 03:43:54 +01:00
|
|
|
focus()
|
2022-09-08 21:40:48 +02:00
|
|
|
}),
|
|
|
|
})
|
|
|
|
)
|
|
|
|
)
|
2021-11-10 18:42:31 +01:00
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
const skippedQuestionsButton = Translations.t.general.skippedQuestions.onClick(() => {
|
|
|
|
skippedQuestions.setData([])
|
|
|
|
})
|
|
|
|
tagsSource.map(
|
|
|
|
(tags) => {
|
|
|
|
if (tags === undefined) {
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
for (let i = 0; i < tagRenderingQuestions.length; i++) {
|
|
|
|
let tagRendering = tagRenderings[i]
|
2021-11-10 18:42:31 +01:00
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
if (skippedQuestions.data.indexOf(i) >= 0) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if (tagRendering.IsKnown(tags)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if (tagRendering.condition) {
|
|
|
|
if (!tagRendering.condition.matchesProperties(tags)) {
|
|
|
|
// Filtered away by the condition, so it is kindof known
|
|
|
|
continue
|
|
|
|
}
|
2021-11-10 18:42:31 +01:00
|
|
|
}
|
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
// this value is NOT known - this is the question we have to show!
|
|
|
|
return i
|
2021-11-10 18:42:31 +01:00
|
|
|
}
|
2022-09-08 21:40:48 +02:00
|
|
|
return undefined // The questions are depleted
|
|
|
|
},
|
|
|
|
[skippedQuestions]
|
|
|
|
)
|
|
|
|
|
|
|
|
const questionsToAsk: Store<BaseUIElement[]> = tagsSource.map(
|
|
|
|
(tags) => {
|
|
|
|
if (tags === undefined) {
|
|
|
|
return []
|
2021-11-10 18:42:31 +01:00
|
|
|
}
|
2022-09-08 21:40:48 +02:00
|
|
|
const qs = []
|
|
|
|
for (let i = 0; i < tagRenderingQuestions.length; i++) {
|
|
|
|
let tagRendering = tagRenderings[i]
|
|
|
|
|
|
|
|
if (skippedQuestions.data.indexOf(i) >= 0) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if (tagRendering.IsKnown(tags)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if (tagRendering.condition && !tagRendering.condition.matchesProperties(tags)) {
|
2022-01-17 21:33:03 +01:00
|
|
|
// Filtered away by the condition, so it is kindof known
|
2022-09-08 21:40:48 +02:00
|
|
|
continue
|
2022-01-17 21:33:03 +01:00
|
|
|
}
|
2021-06-28 00:45:49 +02:00
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
// this value is NOT known - this is the question we have to show!
|
|
|
|
qs.push(tagRenderingQuestions[i])
|
2021-11-12 01:44:13 +01:00
|
|
|
}
|
2022-09-08 21:40:48 +02:00
|
|
|
return qs
|
|
|
|
},
|
|
|
|
[skippedQuestions]
|
|
|
|
)
|
2021-06-28 00:45:49 +02:00
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
super(
|
2023-01-06 03:37:22 +01:00
|
|
|
questionsToAsk.map(
|
|
|
|
(allQuestions) => {
|
|
|
|
const apiState: OsmServiceState = state.osmConnection.apiIsOnline.data
|
|
|
|
if (apiState !== "online" && apiState !== "unknown") {
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
const els: BaseUIElement[] = []
|
|
|
|
if (
|
|
|
|
options.showAllQuestionsAtOnce === true ||
|
|
|
|
options.showAllQuestionsAtOnce["data"]
|
|
|
|
) {
|
|
|
|
els.push(...questionsToAsk.data)
|
|
|
|
} else {
|
|
|
|
els.push(allQuestions[0])
|
|
|
|
}
|
2021-06-28 00:45:49 +02:00
|
|
|
|
2023-01-06 03:37:22 +01:00
|
|
|
if (skippedQuestions.data.length > 0) {
|
|
|
|
els.push(skippedQuestionsButton)
|
|
|
|
}
|
2021-06-28 00:45:49 +02:00
|
|
|
|
2023-01-06 03:37:22 +01:00
|
|
|
return new Combine(els).SetClass("block mb-8")
|
|
|
|
},
|
|
|
|
[state.osmConnection.apiIsOnline]
|
|
|
|
)
|
2021-06-28 00:45:49 +02:00
|
|
|
)
|
|
|
|
|
2022-09-08 21:40:48 +02:00
|
|
|
this.skippedQuestions = skippedQuestions
|
2021-11-12 01:44:13 +01:00
|
|
|
this.restingQuestions = questionsToAsk
|
2023-01-06 03:37:22 +01:00
|
|
|
focus = () => this.ScrollIntoView()
|
2020-10-27 01:01:34 +01:00
|
|
|
}
|
2022-09-08 21:40:48 +02:00
|
|
|
}
|