-
Notifications
You must be signed in to change notification settings - Fork 42
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
Coordinate systems and new coordinate transformations proposal #138
base: main
Are you sure you want to change the base?
Conversation
# Conflicts: # latest/index.bs
* define pixel center coordinate system * add input/output_axes * add transform inverse flag * add affine transform type
* flesh out array space * define array indexing * add more transformation types * start transformation details section and examples * update example
# Conflicts: # latest/index.bs
* use "input" and "output" rather than '*Space' and '*Axes'
* reorder details * clean up table * add rotation * details for sequence * describe inverses * wrap examples
* rephrase matrix storage
* change to "coordinates", removing "Field" * change to "displacements", removing "Field"
* add details for transformation types * (identity, inverseOf, bijection) * describe inputAxes and outputAxes * add new examples
…amples * some clean up
* add mapIndex, mapAXis * add examples * affine stored as flat array only
* sequence does not have by-dimension behavior
* flesh out some examples
Automated Review URLs |
This pull request has been mentioned on Image.sc Forum. There might be relevant details there: https://forum.image.sc/t/ome-ngff-community-call-transforms-and-tables/71792/1 |
unless I'm missing something the "C-vs-F-order-for-affine-transformations" conversation seems to have tailed off without a clear resolution. I suspect this dispute is fundamentally intractable -- whatever indexing order we use, someone will be unhappy. This motivates a solution for representing transformations that can be understood without array indexing conventions. affine transforms without array indexingHere's one idea, probably someone can improve it: {
"transforms" : [
{
"type": "matrix",
"params": {
"z": {"z": 1, "y": 0.5, "x": 0},
"y": {"z": 0.5, "y": 0.2, "x": 0},
"x": {"z" : 0, "y": 1, "x": 1}
}
},
{
"type": "translation",
"params": {"z": 1, "y": 1, "x": 1}
}
]
} This represents a homogeneous affine transformation as a list with two transformations, the first element is a "matrix" transformation, that expresses a linear mapping from named input dimensions to named output dimensions. The second element is a translation transformation, that applies a translation. Both transformations have a {
"type": "translation",
"params": { "z": 1, "y": 1, "x": 1}
} {
"type": "translation",
"params": {"x": 1, "y": 1, "z": 1}
} The "z": {"z": 1, "y": 0.5, "x": 0} The key Personally I find this more readable than the array-of-arrays equivalent, and crucially it completely avoids any array indexing conventions. It also avoids a potentially confusing feature of homogeneous transforms, which is the introduction of an extra dimension in the matrix. Naming the input and output dimensions has other benefits: you can alter the set of keys in e.g., adding output dimensions: {
"type": "matrix",
"params": {
"z": {"z": 1, "y": 0.5, "x": 0},
"y": {"z": 0.5, "y": 0.2, "x": 0},
"x": {"z" : 0, "y": 1, "x": 1}
"phi": {"z" : 1, "y": 1, "x": 1}
}
} or dropping them: {
"type": "matrix",
"params": {
"z": {"z": 1, "y": 0.5, "x": 0},
}
} It doesn't look like a matrix, but that's OK -- a matrix is just one representation of a linear transformation, and the particulars of the matrix representation (which array indexing order? array of arrays or flat array?) is clearly a divisive topic. Honestly, even if everyone uniformly dislikes the JSON I'm proposing above, I think that's a win, because at least everyone is on the same page (provided that the JSON objects I'm proposing can do the same work as a bare array of numbers). making array indexing conventions more explicitEven if we resolve the C vs F order dispute for affine transforms, the spec should be more explicit about the array indexing convention used for other dimensional attributes, like the list of import numpy as np
# place dimension names, or None, at positions in a 1D space
index_reference = [None, 'x', 'y', None]
# when reshaped into a cube, the dimension names end up at the positions
# that depend on the array indexing convention
arr_c = np.array(index_reference).reshape(2,2, order="C")
print(arr_c)
"""
[[None 'x']
['y' None]]
"""
arr_f = np.array(index_reference).reshape(2,2, order="F")
print(arr_f)
"""
[[None 'y']
['x' None]]
"""
# If I use C-ordered indexing, what is the name for the first axis of the data?
arr_c[-1, 0]
"""
'y'
"""
# If i use F ordered indexing, what is the name for the first axis of the data?
arr_f[-1, 0]
"""
'x'
""" Thus, a single piece of JSON-serializable data can unambiguously encode the correct mapping from array indices to dimension names. I think the combination of a) adding this data somewhere in the root of the metadata hierarchy, and b) indexing coordinate transformations with dimension names instead of numbers could go a long way towards smoothing out the C / F divide. |
* changed example axis names to follow current convention * TODO coordinate/displacement field vector order * thanks @thewtex
* fixes rendering using new bikeshed
* including on-disk representation * add an example
* adds section on matrix transformations
* prefer "shape" to "size" for zar arrays
thanks @d-v-b
This PR has four main contributions:
The existing axis specification into
coordinateSystem
s - named collections of axes.coordinateTransformations
now have "input" and "output" coordinate systems.Adds many new useful type of coordinate transformations.
Describes the array/pixel coordinate system (origin at pixel center)
Adds a
longName
field foraxes
See also: