Different thoughts and investingations related to LSF init.
We have 3 UI versions: old (awful), medium (outliner v1), modern (draggable panels)
Flags:
ff_front_1170_outliner_030222_short
— initial FF_1170fflag_feat_front_dev_3873_labeling_ui_improvements_short
— modern FF_3873
Components:
- modern (FF_1170 + FF_3873):
SideTabsPanels
+SidePanels/TabPanels
+OutlinerTree
+BottomBar
- medium (FF_1170):
SidePanels
+OutlinerTree
- old:
SidebarTabs
+AnnotationTab
+Entities/RegionTree
and surprisinglyBottomBar
if FF_3873 enabled without FF_1170
They all have TopBar
with 2 different layouts:
- modern (FF_3873): annotations tab by
AnnotationsCarousel
+ custom actions - medium and old: annotations dropdown by
TopBar/Annotations
+TopBar/Actions
configureStore() has the idea of environment, which comes from development/production run and contains some very important but not irreplaceable things:
- rootElement() (yes, that's a function!) is redundant
- getData() should be embedded
- getExample() is definitely too much — example should be set from outside
- configureApplication() just passes through a lot of methods with default values which are just a placeholders
All annotations should've been always selected on init, most probably because of some weird issues in the past. Don't think we need it now. That could be pretty heavy operation. And it's something we should always think about.
Main flow: LabelStudio -> constructor -> createApp -> configureStore -> initializeStore (+src/Component does the similar)
Also it's called at these places right after assignTask()
:
- AnnotationPreview (?)
- Config/Preview
- ReviewPage -> setTask
- lsf-sdk -> setLSFTask
Should prevent some types of objects from requesting resources too early because the whole LSF will be recreated.
It's default to false
in AppStore;
but set to true
by default in configureStore if value is not reassigned.
Can only be changed on init if Audio3 FF is on.
For some reason it's set to false
in lsf-sdk constructor, but true
in ReviewPage's setTask.
But is set to true
in lsf-sdk setLSFTask.
Apparently it was not used anymore after HumanSignal/label-studio-frontend#1166
wtf?? lot of logic without comments
There is Annotation#exists
getter to check that annotation was saved and can be updated.
userGenerate
is used to mark annotations that were added by user in current session or is still a draft. Is set to true when:
- annotation created by user in
createAnnotation()
- annotation created from prediction in
addAnnotationFromPrediction()
- draft is loaded without annotation in
setAnnotation()
in lsf-sdk - is set by default to true, so it’s worth to check other places; but it defaults to false in
createItem()
Can be submitted in current session, but this flag will remain the same.
called from 3 places:
setLSFTask()
onStorageInitialized()
— called fromAppStore#initializeStore()
, one of the core LSF methods; if there is a task and that’s not a Label Stream it selects annotation based oninitialAnnotation
andlastAnnotation
onDeleteAnnotation()
We have three versions of interface:
- with DEV-1170 FF off we have old interface with
components/SidebarTabs
- with DEV-1170 FF on and DEV-3873 off we have intermediate interface with
components/SidePanels/SidePanels.tsx
- with both FFs on we have new desired interface with
components/SidePanels/TabPanels/SideTabsPanels.tsx
They both are only disabled in couple places.
Also the whole App.panels
prop is only used in the first, the oldest version.
Apparently, panels
were used only to add Comments tab and even this tab was moved into AnnotationTab after
HumanSignal/label-studio-frontend#774. So they are removed along with anything related.
afterReset()
— refill StaredStorage Stores in extender.js
, but looks like it's not used
initRoot — config
for every prediction: addPrediction, selectPrediction, deserializeResults
for every annotation: addAnnotation, selectAnnotation, deserializeResults, reinitHistory (!!)
setInitialValues for last annotation
setHistory to set annotation history — some messed up thing in general
set initialized = true and invoke storageInitialized
They might require annotation to be selected.
ToolManagerMixin.afterAttach() — does lot of things with tools, selecting, unselecting, assigning
RelationStore.afterAttach() — was referring to selected annotation; fixed
Annotation.afterAttach() calls super helpful method annotationAttached(), but it's only implemented in Pairwise
prepareAnnotation -> fixBrokenAnnotation -> some changes on result json
deserializeSingleResult for every result in JSON — some area/result/state manipulations,
should be no side effects.
it calls updateAppearenceFromState
for merged labels and results (only Video tag regions),
but this should only be called on selected annotation.
cleanClassificationAreas — WUT???
updateFromResult for every global classification — a bit redundant unless we select annotation;
but there is hidden
param already, used only on Annotation History deserialization.
deserializeRelation for every relation
- one-way data flow
- describe all possible external things that can affect current view
Still left to fix after HumanSignal/label-studio-frontend#1640
-
Annotation History should be updated with correct annotation when we select it from View All; why not to do the same on usual annotation select? and if we do why do we need special case here?
-
Experiment: use dev addons from outside via additional entry point for webpack. And how to distinguish this from e2e runs? Additional param?
-
description
option should be renamed toinstructions
/~https://github.com/HumanSignal/label-studio-frontend/blob/d0dd212bf7b12cb7f4469c9c2aa32dfe166f5abe/src/stores/RegionStore.js#L181-L183 could be shifted towards self.selection.selectRegions(regions) and then /~https://github.com/HumanSignal/label-studio-frontend/blob/d0dd212bf7b12cb7f4469c9c2aa32dfe166f5abe/src/stores/RegionStore.js#L45-L47 can be changed to usual method called only once.
Only one annotation is selected during task opening.
We have the concept of selecting annotation, when we not just assign it as current one, but also update data in tags, trigger external event, load annotation history, setup hotkeys, set initial values.
Previosly it was required to select every annotation to do some extra work in regions/tags inside it because of some quirky legacy code. I fixed what I found so far and ran all possible tests — they all are green, so consider this change mostly safe.
Benefits:
- improved performance as only one annotation is selected, reducing number of calculations and renders
- no excess network requests to retrieve annotation history for every of these annotations
- more clean code
- one more step to remove one of the oldest legacy concept of
states
: it's not used in init anymore
Things changed:
STORE_INIT_OK
global var added to catch tags that can't properly work with new system (edge cases we missed)- apparently we have only one unit test for tags — Ranker; it's securely mocked
Concerns:
states
in Video tag select only *Labels controls, but changed code is infixBrokenAnnotation
which will do nothing to Video anyway- in potential missed edge case app will just crash
- we could potentially check that we are inside init process by some flag, right?
- removed Predictions panel and related code like showAllPredictions
- removed panels from App and down the stream — we don't use them anywhere (3rd party LSF can be affected, but not sure anyone using panels there)
- converted
LabelStudio.js
to TypeScript - fixed bunch of annoying react warnings (non-html props, wrong values, missed refs)
- started to reorganize our "envs" to get rid of them at some point; for now it's
getRoot()
removed from there - removed
SidebarPage
as it was useless, plus removed multi-tabs support fromSidebarTabs
aspanels
were not used; plus that's all only for the old interface which is about to be removed - removed
hydrated
as it's not used anymore after HumanSignal/label-studio-frontend#1166