Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support multiple segmentation layers #5683

Merged
merged 73 commits into from
Sep 8, 2021
Merged
Show file tree
Hide file tree
Changes from 59 commits
Commits
Show all changes
73 commits
Select commit Hold shift + click to select a range
1dd401c
[WIP] don't crash when multiple segmentation layers exist; start intr…
philippotto Aug 12, 2021
cb08c87
[WIP] fix all flow errors
philippotto Aug 23, 2021
5d31282
[WIP]: implement and use getVisibleSegmentationLayer, getSomeSegmenta…
philippotto Aug 23, 2021
16abedc
[WIP]: implement and use getFirstSegmentationLayer which only needs a…
philippotto Aug 23, 2021
9bd7a88
remove deprecated functions
philippotto Aug 23, 2021
8cd8144
fix wrong parameter
philippotto Aug 23, 2021
1acd9cf
fix isColorLayer
philippotto Aug 23, 2021
c5fbede
adapt shader code to multiple segmentation layers
philippotto Aug 23, 2021
7947f30
fix cell hovering
philippotto Aug 23, 2021
6d4886c
support mappings for multiple segmentation layers
philippotto Aug 24, 2021
1bf928f
fix reducer for initial changes of mapping objects
philippotto Aug 24, 2021
1cafe08
fix runtime exceptions
philippotto Aug 24, 2021
715df28
make shader code and materials compatible with mappings (only one map…
philippotto Aug 24, 2021
4581384
fix some mapping issues
philippotto Aug 24, 2021
422fb67
Merge branch 'master' of github.com:scalableminds/webknossos into mul…
philippotto Aug 25, 2021
c3d36f4
Merge branch 'master' of github.com:scalableminds/webknossos into mul…
philippotto Aug 25, 2021
c738945
fix hovered segment id in status bar
philippotto Aug 25, 2021
74898de
fix multiple mapping bugs (e.g., race condition when activating a map…
philippotto Aug 25, 2021
8228b32
fix wrong attachment of mapping-related textures
philippotto Aug 25, 2021
b036111
fix callHandlerOnSubscribe behavior in listenToStoreProperty if initi…
philippotto Aug 26, 2021
648b24e
update threeJS to 110 so that we can use to avoid discarded updates;…
philippotto Aug 26, 2021
d1c7884
support volume annotation in dataset with multiple segmentation layers
philippotto Aug 26, 2021
e4b1c5a
adapt more volume code to multiple segmentation layers
philippotto Aug 26, 2021
2933c23
remove outdated comment
philippotto Aug 27, 2021
f494249
adapt UI for creation of explorative so that fallback layer can be ch…
philippotto Aug 27, 2021
7f07bfa
change back-end route from boolean withFallback to optional string fa…
philippotto Aug 27, 2021
4d9a3fe
clean up GET parameter construction
philippotto Aug 27, 2021
577b426
further clean up in CreateExplorativeModal logic
philippotto Aug 27, 2021
cc55fb8
fix creation of volume tracing when viewing dataset and only show rem…
philippotto Aug 27, 2021
c21b34c
ensure that only one segmentation layer is visible
philippotto Aug 27, 2021
3d2b6e3
remove activeSegmentationLayerIndex code, as it's not needed anymore
philippotto Aug 27, 2021
6d1d126
trigger agglomeration warning for correct segmentation layer
philippotto Aug 27, 2021
0923f22
adapt data export to multiple segmentation layers
philippotto Aug 27, 2021
0517184
remove unused isRefreshingIsosurfaces store property
philippotto Aug 27, 2021
51d1ed1
fix passing fallbackLayerName when creating explorational
philippotto Aug 30, 2021
fb3ce1e
make mesh management dependent on concrete segmentation layer
philippotto Aug 30, 2021
6d7c811
fix ad-hoc and precomputed mesh usages
philippotto Aug 30, 2021
ae0ed59
rename mesh/isosurface variables to ...ByLayer where appropriate
philippotto Aug 30, 2021
1fb21ca
get rid of getEnforcedSomeSegmentationLayer function
philippotto Aug 30, 2021
1d357ac
add better typing for mesh view and remove some dead code
philippotto Aug 30, 2021
6a84c71
change getResolutionInfoOfSomeSegmentationLayer to getResolutionInfoO…
philippotto Aug 30, 2021
fe793d4
remove some usages of getFirstSegmentationLayer
philippotto Aug 30, 2021
59e58d2
fix default 'Create Annotation' link for fallback layers
philippotto Aug 30, 2021
77a76da
fix linting
philippotto Aug 30, 2021
f0841d5
use correct segmentation layer in version view
philippotto Aug 30, 2021
811b962
remove redundant if-null check
philippotto Aug 30, 2021
bf0af45
make 3-shortcut work better with multiple segmentation layers
philippotto Aug 30, 2021
cc578a7
fix unit tests
philippotto Aug 30, 2021
6727777
improve the api_latest functions
philippotto Aug 30, 2021
5f4466e
further improvements of api_latest
philippotto Aug 30, 2021
1d62c82
format back-end code
philippotto Aug 30, 2021
797abe2
fail if non-existing fallback layer is selected, add fallbackLayerNam…
fm3 Aug 31, 2021
2473b1e
legacy api
fm3 Aug 31, 2021
af27ad9
only allow volume annotation in actual tracing layer
philippotto Aug 31, 2021
9c07917
Merge branch 'multiple-segmentations' of github.com:scalableminds/web…
philippotto Aug 31, 2021
af2fed4
Merge branch 'master' of github.com:scalableminds/webknossos into mul…
philippotto Aug 31, 2021
01d3030
resolve todo affecting features.publicDemoDatasetUrl
philippotto Aug 31, 2021
a701fe5
use currently visible segmentation layer as fallback layer when creat…
philippotto Aug 31, 2021
8bf6bb7
rename activeMapping to activeMappingByLayer
philippotto Aug 31, 2021
fa2b031
fix CI
philippotto Aug 31, 2021
1fc3df6
integrate some PR feedback
philippotto Sep 1, 2021
987ebf4
Apply suggestions from code review
philippotto Sep 1, 2021
d6b8183
rename getRequestedOrVisibleSegmentationLayer, getRequestedOrVisibleS…
philippotto Sep 1, 2021
41a3695
Merge branch 'multiple-segmentations' of github.com:scalableminds/web…
philippotto Sep 1, 2021
d1696b3
integrate further PR feedback
philippotto Sep 2, 2021
15341b5
improve docstrings in api_latest
philippotto Sep 2, 2021
eb4ebf7
Merge branch 'master' of github.com:scalableminds/webknossos into mul…
philippotto Sep 2, 2021
dc69e67
remove other segmentation layers for volume annotations without fallb…
philippotto Sep 6, 2021
3c35e4a
update changelog
philippotto Sep 6, 2021
0b1dd3b
make getVolumeTracingLayerName backwards compatible
philippotto Sep 8, 2021
de2ad94
fix merger mode for hybrids and for skeleton-only annotations
philippotto Sep 8, 2021
a45c81c
fix segmentationOpacity dictated by recommended settings in tasks whe…
philippotto Sep 8, 2021
de5c234
Merge branch 'master' into multiple-segmentations
philippotto Sep 8, 2021
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions MIGRATIONS.unreleased.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,4 +7,6 @@ User-facing changes are documented in the [changelog](CHANGELOG.released.md).

## Unreleased

- For webknossos.org: Change `publicDemoDatasetUrl` in the `features`-block within `application.conf` to be an actionable URL. For example, append `/createExplorative/hybrid?fallbackLayerName=segmentation` to the URL so that a new annotation is created if a user clicks on `Open a Demo Dataset` in the dashboard.

### Postgres Evolutions:
25 changes: 13 additions & 12 deletions app/controllers/AnnotationController.scala
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ import scala.concurrent.ExecutionContext
import scala.concurrent.duration._

case class CreateExplorationalParameters(typ: String,
withFallback: Option[Boolean],
fallbackLayerName: Option[String],
resolutionRestrictions: Option[ResolutionRestrictions])
object CreateExplorationalParameters {
implicit val jsonFormat: OFormat[CreateExplorationalParameters] = Json.format[CreateExplorationalParameters]
Expand Down Expand Up @@ -159,7 +159,7 @@ class AnnotationController @Inject()(
request.identity,
dataSet._id,
tracingType,
request.body.withFallback.getOrElse(true),
request.body.fallbackLayerName,
request.body.resolutionRestrictions.getOrElse(ResolutionRestrictions.empty)
) ?~> "annotation.create.failed"
_ = analyticsService.track(CreateAnnotationEvent(request.identity: User, annotation: Annotation))
Expand All @@ -168,16 +168,17 @@ class AnnotationController @Inject()(
} yield JsonOk(json)
}

def makeHybrid(typ: String, id: String): Action[AnyContent] = sil.SecuredAction.async { implicit request =>
for {
_ <- bool2Fox(AnnotationType.Explorational.toString == typ) ?~> "annotation.makeHybrid.explorationalsOnly"
annotation <- provider.provideAnnotation(typ, id, request.identity)
organization <- organizationDAO.findOne(request.identity._organization)
_ <- annotationService.makeAnnotationHybrid(annotation, organization.name) ?~> "annotation.makeHybrid.failed"
updated <- provider.provideAnnotation(typ, id, request.identity)
json <- annotationService.publicWrites(updated, Some(request.identity)) ?~> "annotation.write.failed"
} yield JsonOk(json)
}
def makeHybrid(typ: String, id: String, fallbackLayerName: Option[String]): Action[AnyContent] =
sil.SecuredAction.async { implicit request =>
for {
_ <- bool2Fox(AnnotationType.Explorational.toString == typ) ?~> "annotation.makeHybrid.explorationalsOnly"
annotation <- provider.provideAnnotation(typ, id, request.identity)
organization <- organizationDAO.findOne(request.identity._organization)
_ <- annotationService.makeAnnotationHybrid(annotation, organization.name, fallbackLayerName) ?~> "annotation.makeHybrid.failed"
updated <- provider.provideAnnotation(typ, id, request.identity)
json <- annotationService.publicWrites(updated, Some(request.identity)) ?~> "annotation.write.failed"
} yield JsonOk(json)
}

def downsample(typ: String, id: String): Action[AnyContent] = sil.SecuredAction.async { implicit request =>
for {
Expand Down
2 changes: 1 addition & 1 deletion app/controllers/LegacyApiController.scala
Original file line number Diff line number Diff line change
Expand Up @@ -147,7 +147,7 @@ class LegacyApiController @Inject()(annotationController: AnnotationController,

def annotationMakeHybrid(typ: String, id: String): Action[AnyContent] = sil.SecuredAction.async { implicit request =>
for {
result <- annotationController.makeHybrid(typ, id)(request)
result <- annotationController.makeHybrid(typ, id, None)(request)
} yield replaceVisibilityInResultJson(result)
}

Expand Down
52 changes: 29 additions & 23 deletions app/models/annotation/AnnotationService.scala
Original file line number Diff line number Diff line change
Expand Up @@ -139,17 +139,23 @@ class AnnotationService @Inject()(
dataSet: DataSet,
dataSource: DataSource,
tracingType: TracingType.Value,
withFallback: Boolean,
fallbackLayerNameOpt: Option[String],
resolutionRestrictions: ResolutionRestrictions,
organizationName: String,
oldTracingId: Option[String] = None)(implicit ctx: DBAccessContext): Fox[(Option[String], Option[String])] = {
def getFallbackLayer: Option[SegmentationLayer] =
if (withFallback) {
dataSource.dataLayers.flatMap {
case layer: SegmentationLayer => Some(layer)
case _ => None
}.headOption
} else None

def getFallbackLayer(fallbackLayerName: String): Fox[SegmentationLayer] =
for {
fallbackLayer <- dataSource.dataLayers
.filter(dl => dl.name == fallbackLayerName)
.flatMap {
case layer: SegmentationLayer => Some(layer)
case _ => None
}
.headOption
.toFox
_ <- bool2Fox(fallbackLayer.elementClass != ElementClass.uint64) ?~> "annotation.volume.uint64"
} yield fallbackLayer

tracingType match {
case TracingType.skeleton =>
Expand All @@ -170,8 +176,7 @@ class AnnotationService @Inject()(
case TracingType.volume =>
for {
client <- tracingStoreService.clientFor(dataSet)
fallbackLayer = getFallbackLayer
_ <- bool2Fox(fallbackLayer.forall(_.elementClass != ElementClass.uint64)) ?~> "annotation.volume.uint64"
fallbackLayer <- Fox.runOptional(fallbackLayerNameOpt)(getFallbackLayer)
volumeTracing <- createVolumeTracing(dataSource,
organizationName,
fallbackLayer,
Expand All @@ -181,8 +186,7 @@ class AnnotationService @Inject()(
case TracingType.hybrid =>
for {
client <- tracingStoreService.clientFor(dataSet)
fallbackLayer = getFallbackLayer
_ <- bool2Fox(fallbackLayer.forall(_.elementClass != ElementClass.uint64)) ?~> "annotation.volume.uint64"
fallbackLayer <- Fox.runOptional(fallbackLayerNameOpt)(getFallbackLayer)
skeletonTracingId <- client.saveSkeletonTracing(
SkeletonTracingDefaults.createInstance.copy(dataSetName = dataSet.name,
editPosition = dataSource.center,
Expand All @@ -199,7 +203,7 @@ class AnnotationService @Inject()(
def createExplorationalFor(user: User,
_dataSet: ObjectId,
tracingType: TracingType.Value,
withFallback: Boolean,
fallbackLayerName: Option[String],
resolutionRestrictions: ResolutionRestrictions)(implicit ctx: DBAccessContext,
m: MessagesProvider): Fox[Annotation] =
for {
Expand All @@ -210,7 +214,7 @@ class AnnotationService @Inject()(
tracingIds <- createTracingsForExplorational(dataSet,
usableDataSource,
tracingType,
withFallback,
fallbackLayerName,
resolutionRestrictions,
organization.name)
teamId <- selectSuitableTeam(user, dataSet) ?~> "annotation.create.forbidden"
Expand All @@ -228,27 +232,29 @@ class AnnotationService @Inject()(
annotation
}

def makeAnnotationHybrid(annotation: Annotation, organizationName: String)(
def makeAnnotationHybrid(annotation: Annotation, organizationName: String, fallbackLayerName: Option[String])(
implicit ctx: DBAccessContext): Fox[Unit] = {
def createNewTracings(dataSet: DataSet, dataSource: DataSource) = annotation.tracingType match {
case TracingType.skeleton =>
createTracingsForExplorational(dataSet,
dataSource,
TracingType.volume,
withFallback = true,
fallbackLayerName,
ResolutionRestrictions.empty,
organizationName).flatMap {
case (_, Some(volumeId)) => annotationDAO.updateVolumeTracingId(annotation._id, volumeId)
case _ => Fox.failure("unexpectedReturn")
}
case TracingType.volume =>
createTracingsForExplorational(dataSet,
dataSource,
TracingType.skeleton,
withFallback = false,
ResolutionRestrictions.empty,
organizationName,
annotation.volumeTracingId).flatMap {
createTracingsForExplorational(
dataSet,
dataSource,
TracingType.skeleton,
fallbackLayerNameOpt = None, // unused when creating skeleton
ResolutionRestrictions.empty,
organizationName,
annotation.volumeTracingId
).flatMap {
case (Some(skeletonId), _) => annotationDAO.updateSkeletonTracingId(annotation._id, skeletonId)
case _ => Fox.failure("unexpectedReturn")
}
Expand Down
2 changes: 1 addition & 1 deletion conf/webknossos.latest.routes
Original file line number Diff line number Diff line change
Expand Up @@ -114,7 +114,7 @@ PUT /annotations/:typ/:id/reset c
PATCH /annotations/:typ/:id/transfer controllers.AnnotationController.transfer(typ: String, id: String)

GET /annotations/:typ/:id/info controllers.AnnotationController.info(typ: String, id: String, timestamp: Long)
PATCH /annotations/:typ/:id/makeHybrid controllers.AnnotationController.makeHybrid(typ: String, id: String)
PATCH /annotations/:typ/:id/makeHybrid controllers.AnnotationController.makeHybrid(typ: String, id: String, fallbackLayerName: Option[String])
PATCH /annotations/:typ/:id/downsample controllers.AnnotationController.downsample(typ: String, id: String)
PATCH /annotations/:typ/:id/unlinkFallback controllers.AnnotationController.unlinkFallback(typ: String, id: String)
DELETE /annotations/:typ/:id controllers.AnnotationController.cancel(typ: String, id: String)
Expand Down
10 changes: 7 additions & 3 deletions frontend/javascripts/admin/admin_rest_api.js
Original file line number Diff line number Diff line change
Expand Up @@ -675,14 +675,14 @@ export function getAnnotationInformation(
export function createExplorational(
datasetId: APIDatasetId,
typ: TracingType,
withFallback: boolean,
fallbackLayerName: ?string,
resolutionRestrictions: ?APIResolutionRestrictions,
options?: RequestOptions = {},
): Promise<APIAnnotation> {
const url = `/api/datasets/${datasetId.owningOrganization}/${datasetId.name}/createExplorational`;
return Request.sendJSONReceiveJSON(
url,
Object.assign({}, { data: { typ, withFallback, resolutionRestrictions } }, options),
Object.assign({}, { data: { typ, fallbackLayerName, resolutionRestrictions } }, options),
);
}

Expand Down Expand Up @@ -761,9 +761,13 @@ export async function importVolumeTracing(tracing: Tracing, dataFile: File): Pro
);
}

export function convertToHybridTracing(annotationId: string): Promise<void> {
export function convertToHybridTracing(
annotationId: string,
fallbackLayerName: ?string,
): Promise<void> {
return Request.receiveJSON(`/api/annotations/Explorational/${annotationId}/makeHybrid`, {
method: "PATCH",
fallbackLayerName,
});
}

Expand Down
Loading