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

Allow dynamic change of natively rendered layer (by re-applying transforms) #7246

Merged
merged 28 commits into from
Aug 14, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
6477e5f
show icon next to layer to indicate whether it is transformed; remove…
philippotto Jul 28, 2023
6df52c1
implement multi key weap map
philippotto Aug 1, 2023
6288d2f
use MultiKeyMap to cache calculation of renderTransforms (only suppor…
philippotto Aug 1, 2023
af77a14
fix linting
philippotto Aug 1, 2023
4791fba
only use c8 in the CI context
philippotto Aug 1, 2023
c826247
implement inversion/chaining of arbitrary transforms; refactor
philippotto Aug 2, 2023
621daff
disable volume annotation when transformation is active for segmentat…
philippotto Aug 2, 2023
e72570d
fix linting
philippotto Aug 2, 2023
9f8a820
test inverse and chaining of affine transforms; refactor
philippotto Aug 2, 2023
40caa98
test inverse and chaining of tps transforms
philippotto Aug 2, 2023
5b8a50e
Merge branch 'master' of github.com:scalableminds/webknossos into tog…
philippotto Aug 8, 2023
a883731
extend and fix tests for chaining mixed affine and tps transforms
philippotto Aug 8, 2023
af189fc
adapt position when toggling transforms
philippotto Aug 8, 2023
bed0baf
adapt zoom when toggling transforms
philippotto Aug 9, 2023
4b98c5f
improve typing of deepIterate
philippotto Aug 9, 2023
b012a48
add scaled TPS test and rename source/target around getPointsC555
philippotto Aug 9, 2023
65d01d5
switch source and target where it was wrong (pure renaming)
philippotto Aug 9, 2023
5491319
further clean up, more tests, more comments
philippotto Aug 9, 2023
c74a040
change source/target order to reduce confusion; fix chaining and add …
philippotto Aug 9, 2023
bc6bbbe
also switch source/target for getPointsC555
philippotto Aug 9, 2023
00c53bf
only show icon when transforms exist in dataset; fix exception when c…
philippotto Aug 10, 2023
bd24828
update changelog
philippotto Aug 10, 2023
482b6ca
also update layer bboxes in 3D viewport when changing natively render…
philippotto Aug 10, 2023
6e68a00
fix linting
philippotto Aug 10, 2023
b4b31f6
integrate transformed-layer icon
philippotto Aug 11, 2023
abbac3a
rename icon to public/images/icon-transformed-layer.svg
philippotto Aug 11, 2023
37f54ee
use different icon for tps and affine transforms
philippotto Aug 11, 2023
e318ec2
Merge branch 'master' into toggle-transforms
philippotto Aug 14, 2023
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
1 change: 1 addition & 0 deletions CHANGELOG.unreleased.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ For upgrade instructions, please check the [migration guide](MIGRATIONS.released
- Added configuration to require users' emails to be verified, added flow to verify emails via link. [#7161](https://github.com/scalableminds/webknossos/pull/7161)
- Added a route to explore and add remote datasets via API. [#7176](https://github.com/scalableminds/webknossos/pull/7176)
- Added option to select multiple segments in the segment list in order to perform batch actions. [#7242](https://github.com/scalableminds/webknossos/pull/7242)
- If a dataset layer is transformed (using an affine matrix or a thin plate spline), it can be dynamically shown without that transform via the layers sidebar. All other layers will be transformed accordingly. [#7246](https://github.com/scalableminds/webknossos/pull/7246)

### Changed
- Small messages during annotating (e.g. “finished undo”, “applying mapping…”) are now click-through so they do not block users from selecting tools. [7239](https://github.com/scalableminds/webknossos/pull/7239)
Expand Down
22 changes: 15 additions & 7 deletions frontend/javascripts/libs/estimate_affine.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
import _ from "lodash";
import { Matrix4x4 } from "mjs";
import { Matrix, solve } from "ml-matrix";
import { Vector3 } from "oxalis/constants";

// Estimates an affine matrix that transforms from source points to target points.
export default function estimateAffine(sourcePoints: Vector3[], targetPoints: Vector3[]) {
// Number of correspondences
const N = sourcePoints.length;
Expand All @@ -11,8 +13,8 @@ export default function estimateAffine(sourcePoints: Vector3[], targetPoints: Ve
const b = [];

for (let i = 0; i < N; i++) {
const q = sourcePoints[i];
const p = targetPoints[i];
const p = sourcePoints[i];
const q = targetPoints[i];
const [px, py, pz] = p;
const [qx, qy, qz] = q;

Expand All @@ -28,11 +30,13 @@ export default function estimateAffine(sourcePoints: Vector3[], targetPoints: Ve
const xMatrix = solve(A, b);
const x = xMatrix.to1DArray();
const error = Matrix.sub(b, new Matrix(A).mmul(xMatrix)).to1DArray();
console.log(
"Affine estimation error: ",
error,
`(mean=${_.mean(error.map((el) => Math.abs(el)))})`,
);
if (!process.env.IS_TESTING) {
console.log(
"Affine estimation error: ",
error,
`(mean=${_.mean(error.map((el) => Math.abs(el)))})`,
);
}

const affineMatrix = new Matrix([
[x[0], x[1], x[2], x[3]],
Expand All @@ -43,3 +47,7 @@ export default function estimateAffine(sourcePoints: Vector3[], targetPoints: Ve

return new Matrix(affineMatrix);
}

export function estimateAffineMatrix4x4(sourcePoints: Vector3[], targetPoints: Vector3[]) {
return estimateAffine(sourcePoints, targetPoints).to1DArray() as any as Matrix4x4;
}
3 changes: 3 additions & 0 deletions frontend/javascripts/libs/mjs.ts
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,9 @@ const M4x4 = {
// and also returns Array<Vector3>
transformVectorsAffine(m: Matrix4x4, _points: Vector3[]): Vector3[] {
const points: Array<Array<number>> = _points as any as Array<Array<number>>;
if (!Array.isArray(_points[0])) {
throw new Error("transformVectorsAffine doesn't support typed arrays at the moment.");
}
// @ts-ignore
return chunk3(M4x4.transformPointsAffine(m, _.flatten(points)));
},
Expand Down
35 changes: 35 additions & 0 deletions frontend/javascripts/libs/multi_key_map.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
type RecursiveMap<K, V> = Map<K, V | RecursiveMap<K, V>>;

export default class MultiKeyMap<E, V, K extends E[]> {
map: RecursiveMap<E, V> = new Map<E, RecursiveMap<E, V>>();
set(keys: K, value: V) {
let currentMap = this.map;
for (const [index, key] of keys.entries()) {
if (index < keys.length - 1) {
let foundMap = currentMap.get(key) as RecursiveMap<E, V> | undefined;
if (foundMap == null) {
foundMap = new Map<E, RecursiveMap<E, V>>();
currentMap.set(key, foundMap);
}
currentMap = foundMap;
} else {
currentMap.set(key, value);
}
}
}
get(keys: K): V | undefined {
let currentMap = this.map;
for (const [index, key] of keys.entries()) {
if (index < keys.length - 1) {
const foundMap = currentMap.get(key);
if (foundMap === undefined) {
return undefined;
}
currentMap = foundMap as RecursiveMap<E, V>;
} else {
return currentMap.get(key) as V | undefined;
}
}
throw new Error("MultiKeyMap.get called with empty key.");
}
}
23 changes: 21 additions & 2 deletions frontend/javascripts/libs/thin_plate_spline.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import _ from "lodash";
import { Matrix, solve } from "ml-matrix";
import { Vector3 } from "oxalis/constants";
import { V3 } from "./mjs";

class TPS1d {
// This class accepts 3-dimensional control points
Expand Down Expand Up @@ -110,13 +111,23 @@ export default class TPS3D {
tpsY = new TPS1d();
tpsZ = new TPS1d();

constructor(sourcePoints: Vector3[], targetPoints: Vector3[]) {
unscaledSourcePoints: Vector3[];
unscaledTargetPoints: Vector3[];
scale: Vector3;

constructor(unscaledSourcePoints: Vector3[], unscaledTargetPoints: Vector3[], scale: Vector3) {
const sourcePoints = unscaledSourcePoints.map((point) => V3.scale3(point, scale, [0, 0, 0]));
const targetPoints = unscaledTargetPoints.map((point) => V3.scale3(point, scale, [0, 0, 0]));

const [cps, offsetX, offsetY, offsetZ] = this.getControlPointsWithOffsets(
sourcePoints,
targetPoints,
true,
);

this.unscaledSourcePoints = unscaledSourcePoints;
this.unscaledTargetPoints = unscaledTargetPoints;
this.scale = scale;

this.tpsX.fit(offsetX, cps);
this.tpsY.fit(offsetY, cps);
this.tpsZ.fit(offsetZ, cps);
Expand All @@ -129,6 +140,14 @@ export default class TPS3D {
return [x + dx, y + dy, z + dz];
}

transformUnscaled(x: number, y: number, z: number): Vector3 {
// Scale, transform and unscale input.
const scaled = V3.scale3([x, y, z], this.scale, [0, 0, 0]);
const scaledTransformed = this.transform(...scaled);
const unscaledTransformed = V3.divide3(scaledTransformed, this.scale, [0, 0, 0]);
return unscaledTransformed;
}

getControlPointsWithOffsets(
sourcePoints: Vector3[],
targetPoints: Vector3[],
Expand Down
21 changes: 21 additions & 0 deletions frontend/javascripts/libs/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1149,3 +1149,24 @@ export function minValue(array: Array<number>): number {
}
return value;
}

/*
* Iterates over arbitrary objects recursively and calls the callback function.
*/
type Obj = Record<string, unknown>;
export const deepIterate = (obj: Obj | Obj[] | null, callback: (val: unknown) => void) => {
if (obj == null) {
return;
}
const items = Array.isArray(obj) ? obj : Object.values(obj);
items.forEach((item) => {
callback(item);

if (typeof item === "object") {
// We know that item is an object or array which matches deepIterate's signature.
// However, TS doesn't infer this.
// @ts-ignore
deepIterate(item, callback);
}
});
};
14 changes: 11 additions & 3 deletions frontend/javascripts/oxalis/api/api_latest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ import {
getResolutionInfo,
getVisibleSegmentationLayer,
getMappingInfo,
flatToNestedMatrix,
} from "oxalis/model/accessors/dataset_accessor";
import {
getPosition,
Expand Down Expand Up @@ -2038,7 +2039,7 @@ class DataApi {
*
* @example
*
* api.data._setLayerTransforms(
* api.data._setAffineLayerTransforms(
* "C555_DIAMOND_2f",
* new Float32Array([
* 0.03901274364025348, -0.08498337289603758, 0.00782446404039791, 555.7948181512004,
Expand All @@ -2048,8 +2049,15 @@ class DataApi {
* ]),
* );
*/
_setLayerTransforms(layerName: string, transforms: Matrix4x4) {
Store.dispatch(setLayerTransformsAction(layerName, Array.from(transforms) as Vector16));
_setAffineLayerTransforms(layerName: string, transforms: Matrix4x4) {
const coordinateTransforms = [
{
type: "affine" as const,
matrix: flatToNestedMatrix(Array.from(transforms) as Vector16),
},
];

Store.dispatch(setLayerTransformsAction(layerName, coordinateTransforms));
}

/*
Expand Down
16 changes: 12 additions & 4 deletions frontend/javascripts/oxalis/controller/scene_controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,6 @@ import { getVoxelPerNM } from "oxalis/model/scaleinfo";
import { Model } from "oxalis/singletons";
import type { OxalisState, SkeletonTracing, UserBoundingBox } from "oxalis/store";
import Store from "oxalis/store";
import type { APIDataLayer } from "types/api_flow_types";
import SegmentMeshController from "./segment_mesh_controller";

const CUBE_COLOR = 0x999999;
Expand Down Expand Up @@ -441,9 +440,10 @@ class SceneController {
this.rootNode.add(this.userBoundingBoxGroup);
}

setLayerBoundingBoxes(layers: APIDataLayer[]): void {
updateLayerBoundingBoxes(): void {
const state = Store.getState();
const dataset = state.dataset;
const layers = getDataLayers(dataset);

const newLayerBoundingBoxGroup = new THREE.Group();
this.layerBoundingBoxes = Object.fromEntries(
Expand All @@ -458,7 +458,11 @@ class SceneController {
isHighlighted: false,
});
bbCube.getMeshes().forEach((mesh) => {
const transformMatrix = getTransformsForLayerOrNull(dataset, layer)?.affineMatrix;
const transformMatrix = getTransformsForLayerOrNull(
dataset,
layer,
state.datasetConfiguration.nativelyRenderedLayerName,
)?.affineMatrix;
if (transformMatrix) {
const matrix = new THREE.Matrix4();
// @ts-ignore
Expand Down Expand Up @@ -550,7 +554,11 @@ class SceneController {
);
listenToStoreProperty(
(storeState) => getDataLayers(storeState.dataset),
(layers) => this.setLayerBoundingBoxes(layers),
() => this.updateLayerBoundingBoxes(),
);
listenToStoreProperty(
(storeState) => storeState.datasetConfiguration.nativelyRenderedLayerName,
() => this.updateLayerBoundingBoxes(),
);
listenToStoreProperty(
(storeState) => getSomeTracing(storeState.tracing).boundingBox,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import {
getTransformsForLayer,
getResolutionInfoByLayer,
getResolutionInfo,
getTransformsPerLayer,
} from "oxalis/model/accessors/dataset_accessor";
import {
getActiveMagIndicesForLayers,
Expand Down Expand Up @@ -226,6 +227,9 @@ class PlaneMaterialFactory {
this.uniforms.activeMagIndices = {
value: Object.values(activeMagIndices),
};
const nativelyRenderedLayerName =
Store.getState().datasetConfiguration.nativelyRenderedLayerName;
const dataset = Store.getState().dataset;
for (const dataLayer of Model.getAllLayers()) {
const layerName = sanitizeName(dataLayer.name);

Expand All @@ -241,14 +245,18 @@ class PlaneMaterialFactory {
this.uniforms[`${layerName}_unrenderable`] = {
value: 0,
};
const dataset = Store.getState().dataset;
const layer = getLayerByName(dataset, dataLayer.name);

this.uniforms[`${layerName}_transform`] = {
value: invertAndTranspose(getTransformsForLayer(dataset, layer).affineMatrix),
value: invertAndTranspose(
getTransformsForLayer(dataset, layer, nativelyRenderedLayerName).affineMatrix,
),
};
this.uniforms[`${layerName}_has_transform`] = {
value: !_.isEqual(getTransformsForLayer(dataset, layer).affineMatrix, Identity4x4),
value: !_.isEqual(
getTransformsForLayer(dataset, layer, nativelyRenderedLayerName).affineMatrix,
Identity4x4,
),
};
}

Expand Down Expand Up @@ -431,8 +439,9 @@ class PlaneMaterialFactory {
// isn't relevant which is why it can default to [1, 1, 1].

let representativeMagForVertexAlignment: Vector3 = [Infinity, Infinity, Infinity];
const state = Store.getState();
for (const [layerName, activeMagIndex] of Object.entries(activeMagIndices)) {
const layer = getLayerByName(Store.getState().dataset, layerName);
const layer = getLayerByName(state.dataset, layerName);
const resolutionInfo = getResolutionInfo(layer.resolutions);
// If the active mag doesn't exist, a fallback mag is likely rendered. Use that
// to determine a representative mag.
Expand All @@ -443,7 +452,11 @@ class PlaneMaterialFactory {
: null;

const hasTransform = !_.isEqual(
getTransformsForLayer(Store.getState().dataset, layer).affineMatrix,
getTransformsForLayer(
state.dataset,
layer,
state.datasetConfiguration.nativelyRenderedLayerName,
).affineMatrix,
Identity4x4,
);
if (!hasTransform && suitableMag) {
Expand Down Expand Up @@ -780,27 +793,37 @@ class PlaneMaterialFactory {

this.storePropertyUnsubscribers.push(
listenToStoreProperty(
(storeState) => storeState.dataset.dataSource.dataLayers,
(layers) => {
(storeState) =>
getTransformsPerLayer(
storeState.dataset,
storeState.datasetConfiguration.nativelyRenderedLayerName,
),
(transformsPerLayer) => {
this.scaledTpsInvPerLayer = {};
const state = Store.getState();
const layers = state.dataset.dataSource.dataLayers;
for (let layerIdx = 0; layerIdx < layers.length; layerIdx++) {
const layer = layers[layerIdx];
const name = sanitizeName(layer.name);
const transforms = getTransformsForLayer(Store.getState().dataset, layer);
const transforms = transformsPerLayer[layer.name];
const { affineMatrix } = transforms;
const scaledTpsInv =
transforms.type === "thin_plate_spline" ? transforms.scaledTpsInv : null;

if (scaledTpsInv) {
this.scaledTpsInvPerLayer[name] = scaledTpsInv;
} else {
delete this.scaledTpsInvPerLayer[name];
}

this.uniforms[`${name}_transform`].value = invertAndTranspose(affineMatrix);
const hasTransform = !_.isEqual(affineMatrix, Identity4x4);
console.log(`${name}_has_transform`, hasTransform);
this.uniforms[`${name}_has_transform`] = {
value: hasTransform,
};
}
this.recomputeShaders();
},
true,
),
Expand Down Expand Up @@ -855,6 +878,9 @@ class PlaneMaterialFactory {
}

recomputeShaders = _.throttle(() => {
if (this.material == null) {
return;
}
const [newFragmentShaderCode, additionalUniforms] = this.getFragmentShaderWithUniforms();
for (const [name, value] of Object.entries(additionalUniforms)) {
this.uniforms[name] = value;
Expand Down
Loading