-
Notifications
You must be signed in to change notification settings - Fork 29
Using arrays
PyGLM's array type was introduced in version 2.0.0 to reduce the likelihood of requiring users to also use numpy besides glm.
It's mainly intended to provide a way of passing multiple glm type instances (such as vectors) to external C functions
(such as glBufferData).
PyGLM's arrays are pure data copies of one or multiple instances of a single PyGLM type.
For example, an array could copy the data of five different vec3
instances.
However, it cannot copy the data of instances, that don't have the same type, like two vec2
instances and one dvec2
instance.
Additionally, the data inside the array is only a copy, thus if the data inside the array is modified, it won't affect the instances it copied the data from.
Arrays can be initialized in a few different ways.
An array can be initialized with any number of vectors, metrices or quaternions, as long as they're all of the same type.
>>> array(vec3(1, 2, 3), vec3(4, 5, 6))
array(vec3(1, 2, 3), vec3(4, 5, 6))
>>> array(vec3(), ivec3())
TypeError: arrays have to be initialized with arguments of the same glm type
The same holds true for ctypes numbers:
>>> array(int8(2), int8(3))
array(c_int8(2), c_int8(3))
>>> array(int8(2), int16(3))
TypeError: arrays have to be initialized with arguments of the same type
You can also create an array of ctypes numbers from normal numbers using from_numbers
, which needs to have the data type as it's first argument:
>>> array.from_numbers(int8, 2, 3)
array(c_int8(2), c_int8(3))
>>> array(int8, 2, 3) # alternative way
array(c_int8(2), c_int8(3))
Note: The list representations of vecs, mats and quats such as ((1, 2), (3, 4))
(an alias for mat2(1, 2, 3, 4)
) cannot be used here.
You can obtain a copy of an array by using the copy constructor.
arr_copy = array(arr)
You can convert any compatible type to a PyGLM array.
This includes lists, tuples, etc. and types that support the buffer protocol (such as numpy's arrays):
>>> array((ivec1(), ivec1()))
array(ivec1(0), ivec1(0))
>>> array([dmat2()])
array(dmat2x2((1, 0), (0, 1)))
>>> array(numpy.array([[1,2,3]]))
array(ivec3(1, 2, 3))
>>> array([[1,2,3]])
TypeError: invalid argument type(s) for array()
Note: array buffers that store length 4 items are interpreted as vec4s rather than quats.
If you don't need or don't want a copy of an array or buffer, but want a reference instead (i.e. use the same data in memory as another array / buffer), you can do so by using glm.array.as_reference
.
>>> arr = array(vec3(1))
>>> arr2 = array.as_reference(arr)
>>> arr == arr2
True
>>> arr.address == arr2.address
True
>>> arr[0] = vec3(2) # if you change one of them, the other changes as well
>>> arr2
array(vec3(2, 2, 2))
Note: as_reference
only works with array instances or buffers (e.g. numpy.array
).
Also it may not always be possible to create a reference copy, in which case a normal copy is made and a warning is raised.
You can initialize an array with any given number of zeros or a given type:
>>> array.zeros(4, uint8)
array(c_uint8(0), c_uint8(0), c_uint8(0), c_uint8(0))
>>> array.zeros(2, vec3)
array(vec3(0, 0, 0), vec3(0, 0, 0))
You can initialize an array with numbers and a (ctypes) data type using glm.array.from_numbers
:
>>> array.from_numbers(int8, 1, 2, 3)
array(c_int8(1), c_int8(2), c_int8(3))
>>> array.from_numbers(float32, 4.2, 1.1)
array(c_float(4.2), c_float(1.1))
>>> array(int8, 1, 2, 3) # You can also use the array() constructor, but beware that the dedicated function is faster
array(c_int8(1), c_int8(2), c_int8(3))
>>> array.from_numbers(vec1, 1, 2, 3)
TypeError: Invalid argument type for from_number(), expected a ctypes data type as the first argument. Got 'type'
PyGLM arrays have the following members:
Name | Type | Description |
---|---|---|
element_type | type | Type class of the contained elements (e.g. glm.vec3 ) |
length | int | Number of elements contained by a given array |
address | int | The memory address where an array's data is stored |
ptr | c_void_p | A ctypes pointer that points to the content of an array |
nbytes | int | The total data size in bytes |
typecode | str | A single character, describing the data type of the elements' values, according to this list |
dtype | str | A numpy-like data type string |
ctype | str | The respective ctypes data type |
itemsize | int | The size of one array element in bytes |
dt_size | int | The size of each single component of the elements in bytes (size of data type) |
readonly | int | Whether or not the array is read-only |
reference | int | The reference to the array owning the data (if any) |
Arrays support the copy protocol (see here).
You can use copy.copy(<array>)
or copy.deepcopy(<array>)
to get a copy of an array.
Arrays support pickling (as of PyGLM 2.0.0), which is Python's serialization method.
Any array has a to_list()
and a to_tuple()
function, which return's the arrays's data represented as a list or tuple respectively.
Any array has a to_bytes()
and a static from_bytes()
method, which allows for conversion of the array's data to and from bytes strings.
The from_bytes()
method takes the bytes string and a target type (uint8
by default) as arguments.
Example:
>>> array(uint8(1), uint8(2)).to_bytes()
b'\x01\x02'
>>> array.from_bytes(b"\x01\x02\x03", uint8)
array(c_uint8(1), c_uint8(2), c_uint8(3))
>>> array.from_bytes(b"\x00\x00\x00\x00", vec1)
array(vec1(0))
>>> array.from_bytes(b"\x00\x00\x00\x00" * 8, vec1)
array(vec1(0), vec1(0), vec1(0), vec1(0), vec1(0), vec1(0), vec1(0), vec1(0))
The array class has a static from_numbers
method, which allows for creation of a one-dimensional array of numbers.
It takes a ctypes number type as it's first argument.
Example:
>>> array.from_numbers(c_float, 1.2, 3.4)
array(c_float(1.2), c_float(3.4))
>>> array.from_numbers(int32, 1, 3, 4, 5)
array(c_int32(1), c_int32(3), c_int32(4), c_int32(5))
The array class also has a static as_reference
method, which allows for creation of a reference copy of other arrays or objects that support the buffer protocol.
A reference copy means that the newly created array will use the same data in memory as the source array.
It will also keep a reference to the object it shares the data with in the reference
member.
Example:
>>> arr = array(vec2(1))
>>> arr2 = array.as_reference(arr)
>>> arr == arr2
True
>>> arr is arr2.reference
True
>>> arr[0] = vec2(-1) # if you change one of them, the other changes as well
>>> arr2
array(vec2(-1, -1))
Additionally, the array class has a static zeros
method, which allows for creation of an array with items that are initialized with zeros.
This is the fastest way of creating an array, as it uses the builtin calloc
function to allocate the memory and initialize it in the same step.
Example:
>>> array.zeros(4, uint8)
array(c_uint8(0), c_uint8(0), c_uint8(0), c_uint8(0))
>>> array.zeros(2, vec3)
array(vec3(0, 0, 0), vec3(0, 0, 0))
>>> array.zeros(1, mat4)
array(mat4x4((0, 0, 0, 0), (0, 0, 0, 0), (0, 0, 0, 0), (0, 0, 0, 0)))
You can filter an array using a custom filtering function (i.e. create a new array with all of this array's elements that match a certain criteria).
The filtering function is called with each element of the array and must return True
for the elements to keep and False
for the elements to discard.
Example:
>>> array(c_float, 1, 2, 3, 4, 5).filter(lambda x: x > 3)
array(c_float(4), c_float(5))
>>> array(vec3(1), vec3(2), vec3(3)).filter(lambda x: x.x + x.y == 2)
array(vec3(1, 1, 1))
You can map the elements of an array to one or any number of new values using a custom mapping function.
The mapping function is supplied with the amounts of arguments that the map function is given.
If the mapping function returns None
, the element is discarded (much like filter).
If it returns a single value, that value becomes a new element of the resulting array.
If it returns a tuple, all of the tuple's items are added to the resulting array.
When the mapping function returns numbers, they are interpreted as ctypes numbers of the same type as the array that map
was called on.
If you want to specify a custom ctypes type to use, there is a keyword argument called ctype
that can be used to do that.
Example:
>>> array(c_float, 1, 2, 3).map(lambda x: x + 1)
array(c_float(2), c_float(3), c_float(4))
>>> array(c_float, 1, 2, 3).map(vec3)
array(vec3(1, 1, 1), vec3(2, 2, 2), vec3(3, 3, 3))
>>> array(c_float, 1, 2, 3, 4, 5).map(lambda x: x if x > 3 else None)
array(c_float(4), c_float(5))
>>> array(c_float, 1, 2).map(lambda x: (x, x))
array(c_float(1), c_float(1), c_float(2), c_float(2))
>>> array(c_float, 1, 2).map(lambda x: (3, 4) if x == 1 else 5)
array(c_float(3), c_float(4), c_float(5))
>>> arr1 = array(c_float, 1, 2, 3)
>>> arr2 = array(c_float, 4, 5, 6)
>>> arr1.map(lambda x, y: (x + x) * y, arr2)
array(c_float(8), c_float(20), c_float(36))
>>> arr3 = array(c_float, 7, 8, 9)
>>> arr1.map(lambda x, y, z: vec3(x, y, z), arr2, arr3)
array(vec3(1, 4, 7), vec3(2, 5, 8), vec3(3, 6, 9))
>>> arr1.map(lambda x, y, z: x * y + z, arr2, arr3)
array(c_float(11), c_float(18), c_float(27))
>>> arr4 = array(vec3(1, 2, 3), vec3( 4, 5, 6))
>>> arr4.map(normalize)
array(vec3(0.267261, 0.534522, 0.801784), vec3(0.455842, 0.569803, 0.683764))
>>> arr5 = array(vec3(7, 8, 9), vec3(10,11,12))
>>> arr4.map(dot, arr5)
array(c_float(50), c_float(167))
>>> arr4.map(dot, arr5, ctype = c_int32)
array(c_int32(50), c_int32(167))
You can sort an array by using a custom sorting function.
The sorting function is called with two elements from the array and should return -1
if the first element comes before the second element in order.
Otherwise it should return 0
for equal elements and 1
if the first element comes after the second.
The sorting algorithm used is a recursive quicksort.
Example:
>>> arr = array(c_float, 6, 5, 4, 3, 2, 1)
>>> arr.sort(lambda x, y: -1 if x < y else 0 if x == y else 1)
>>> arr
array(c_float(1), c_float(2), c_float(3), c_float(4), c_float(5), c_float(6))
>>> arr.sort(lambda x, y: int(sign(y - x)))
>>> arr
array(c_float(6), c_float(5), c_float(4), c_float(3), c_float(2), c_float(1))
>>> arr.sort(cmp) # using glm.cmp
>>> arr
array(c_float(1), c_float(2), c_float(3), c_float(4), c_float(5), c_float(6))
You can split an array into the components of it's elements using the split_components()
method.
Splits each element of this array into it's components.
Returns one or multiple arrays wrapped in a tuple.
Example:
>>> arr = array(vec3(1, 2, 3), vec3(4, 5, 6))
>>> arr.split_components()
(array(c_float(1), c_float(4)),
array(c_float(2), c_float(5)),
array(c_float(3), c_float(6)))
>>> arr = array(mat2((1,2),(3,4)), mat2((5,6),(7,8)), mat2((9,10),(11,12)))
>>> arr.split_components()
(array(vec2(1, 2), vec2(5, 6), vec2(9, 10)),
array(vec2(3, 4), vec2(7, 8), vec2(11, 12)))
>>> arr = array(c_float, 6, 5, 4, 3, 2, 1) # doesn't have any components
>>> arr.split_components()
NotImplementedError: split_components() is not defined for ctypes arrays
Used to apply a binary function to this array's elements cumulatively, reducing the array to a single value. If an optional initializer is given, it is placed before the first element.
Example:
>>> arr = array(c_float, 6, 5, 4, 3, 2, 1)
>>> arr.reduce(lambda x, y: x + y) # (((((6+5)+4)+3)+2)+1)
21.0
>>> arr.reduce(mul) # (((((6*5)*4)*3)*2)*1)
720.0
>>> arr.reduce(sub) # (((((6-5)-4)-3)-2)-1)
-9.0
>>> arr.reduce(sub, 21) # with initializer: ((((((21-6)-5)-4)-3)-2)-1)
0.0
Arrays can be combined / concatenated using the concat()
method, as long as they have the same element type.
>>> array(vec2(1, 2)).concat(array(vec2(3, 4)))
array(vec2(1, 2), vec2(3, 4))
>>> array(vec4()).concat(array(vec1()))
ValueError: the given arrays are incompatible
Arrays can be repeated a given number of times using the repeat()
method.
>>> array(vec3(1, 2, 3)).repeat(3)
array(vec3(1, 2, 3), vec3(1, 2, 3), vec3(1, 2, 3))
You can reinterpret the data of an array as a different element type using the reinterpret_cast()
method.
Example:
>>> array(vec3(1, 2, 3)).reinterpret_cast(float32)
array(c_float(1), c_float(2), c_float(3))
>>> array.from_numbers(float32, 1, 2, 3, 4, 5, 6, 7, 8, 9).reinterpret_cast(vec3)
array(vec3(1, 2, 3), vec3(4, 5, 6), vec3(7, 8, 9))
Arrays support a dozen numeric operations:
- Addition (
+
) - Subtraction (
-
) - Multiplication (
*
) - Division (
/
) - Modulus (
%
) - Power (
**
) - Negation (
-
) - Absolution (
abs()
) - Inversion (
~
) - Left shift (
<<
) - Right shift (
>>
) - Bitwise and (
&
) - Bitwise or (
|
) - Bitwise xor (
^
)
Note: Not all types are compatible though.
You can access the individual elements of an array using indices.
Likewise you can also modify it's data by overwriting it or delete it using del
>>> arr = array(vec1(1), vec1(2), vec1(3))
>>> arr[0]
vec1( 1 )
>>> arr[1] = vec1(0)
>>> arr
array(vec1(1), vec1(0), vec1(3))
>>> del arr[2]
>>> arr
array(vec1(1), vec1(0))
You can also use slices to get or modify sub-arrays:
>>> arr = array(vec1(1), vec1(2), vec1(3), vec1(4))
>>> arr[:2]
array(vec1(1), vec1(2))
>>> arr[::2]
array(vec1(1), vec1(3))
>>> del arr[1:3]
>>> arr
array(vec1(1), vec1(4))
>>> arr[:] = array(vec1(8), vec1(9))
>>> arr
array(vec1(8), vec1(9))
Slices have the following syntax: start_index : stop_index : optional_step
, meaning you start at start
and go step
steps until you've reached or passed stop
(exclusive) or the greatest possible index.
You can check wether or not an element is present in the array using the in
operator.
>>> arr = array(vec2(1, 2), vec2(3, 4))
>>> vec2() in arr
False
>>> vec2(3, 4) in arr
True
>>> vec2(2, 3) in arr
False
You can acquire the length of an array using the built-in len()
function.
>>> len(array(vec1(0), vec1(1)))
2
>>> len(array(vec2(0, 1), vec2(2, 3), vec2(4, 5)))
3
You can get a string representation of an array using the built-in str()
function.
Example:
>>> print(str(array(vec2(1, 2), vec2(3, 4), vec2(5, 6))))
[
[ 1, 2 ],
[ 3, 4 ],
[ 5, 6 ],
]
>>> print(str(array(mat2(1, 2, 3, 4), mat2(5, 6, 7, 8))))
[
[
[ 1, 2 ],
[ 3, 4 ],
],
[
[ 5, 6 ],
[ 7, 8 ],
],
]
You can get a reproducable string representation of an array using the built-in str()
function.
Example:
>>> print(repr(array(vec2(1, 2), vec2(3, 4), vec2(5, 6))))
array(vec2(1, 2), vec2(3, 4), vec2(5, 6))
>>> print(repr(array(mat2(1, 2, 3, 4), mat2(5, 6, 7, 8))))
array(mat2x2((1, 2), (3, 4)), mat2x2((5, 6), (7, 8)))
You can get an iterator from an array using iter()
>>> arr = array(vec2(1, 2), vec2(3, 4))
>>> it = iter(arr)
>>> next(it)
vec2( 1, 2 )
>>> next(it)
vec2( 3, 4 )
You can generate a hash value for arrays using hash()
Example:
>>> arr = array(vec3(1), vec3(2), vec3(3), vec3(4))
>>> hash(arr)
-2624592468369027458
>>> arr2 = array(vec3(1), vec3(2), vec3(3))
>>> hash(arr2)
9163283608357050393
>>> arr3 = arr2 + array(vec3(4))
>>> hash(arr3)
-2624592468369027458