2014-02-10 02:10:30 +01:00
/*************************************************************************/
/* array.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 14:16:55 +02:00
/* https://godotengine.org */
2014-02-10 02:10:30 +01:00
/*************************************************************************/
2020-01-01 11:16:22 +01:00
/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md). */
2014-02-10 02:10:30 +01:00
/* */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the */
/* "Software"), to deal in the Software without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of the Software, and to */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions: */
/* */
/* The above copyright notice and this permission notice shall be */
/* included in all copies or substantial portions of the Software. */
/* */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/*************************************************************************/
2018-01-05 00:50:27 +01:00
2014-02-10 02:10:30 +01:00
# include "array.h"
2017-01-16 08:04:19 +01:00
2020-04-21 00:06:00 +02:00
# include "container_type_validate.h"
2020-11-07 23:33:38 +01:00
# include "core/object/class_db.h"
# include "core/object/script_language.h"
# include "core/templates/hashfuncs.h"
# include "core/templates/vector.h"
# include "core/variant/variant.h"
2014-02-10 02:10:30 +01:00
2018-03-09 20:16:08 +01:00
class ArrayPrivate {
public :
2014-02-10 02:10:30 +01:00
SafeRefCount refcount ;
Vector < Variant > array ;
2020-04-21 00:06:00 +02:00
ContainerTypeValidate typed ;
2014-02-10 02:10:30 +01:00
} ;
2017-03-05 16:44:50 +01:00
void Array : : _ref ( const Array & p_from ) const {
2014-02-10 02:10:30 +01:00
ArrayPrivate * _fp = p_from . _p ;
ERR_FAIL_COND ( ! _fp ) ; // should NOT happen.
2020-05-14 16:41:43 +02:00
if ( _fp = = _p ) {
2017-09-22 05:58:29 +02:00
return ; // whatever it is, nothing to do here move along
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
bool success = _fp - > refcount . ref ( ) ;
2017-09-22 05:58:29 +02:00
ERR_FAIL_COND ( ! success ) ; // should really not happen either
2014-02-10 02:10:30 +01:00
_unref ( ) ;
2017-01-11 12:53:31 +01:00
_p = p_from . _p ;
2014-02-10 02:10:30 +01:00
}
void Array : : _unref ( ) const {
2020-05-14 16:41:43 +02:00
if ( ! _p ) {
2014-02-10 02:10:30 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
if ( _p - > refcount . unref ( ) ) {
memdelete ( _p ) ;
}
2020-04-02 01:20:12 +02:00
_p = nullptr ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
Variant & Array : : operator [ ] ( int p_idx ) {
2018-07-25 03:11:03 +02:00
return _p - > array . write [ p_idx ] ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
const Variant & Array : : operator [ ] ( int p_idx ) const {
2014-02-10 02:10:30 +01:00
return _p - > array [ p_idx ] ;
}
int Array : : size ( ) const {
return _p - > array . size ( ) ;
}
2020-05-14 14:29:06 +02:00
2020-12-15 13:04:21 +01:00
bool Array : : is_empty ( ) const {
return _p - > array . is_empty ( ) ;
2014-02-10 02:10:30 +01:00
}
2020-05-14 14:29:06 +02:00
2014-02-10 02:10:30 +01:00
void Array : : clear ( ) {
_p - > array . clear ( ) ;
}
2017-03-05 16:44:50 +01:00
bool Array : : operator = = ( const Array & p_array ) const {
return _p = = p_array . _p ;
2014-02-10 02:10:30 +01:00
}
2020-11-05 03:01:55 +01:00
bool Array : : operator ! = ( const Array & p_array ) const {
return ! operator = = ( p_array ) ;
}
bool Array : : operator < ( const Array & p_array ) const {
int a_len = size ( ) ;
int b_len = p_array . size ( ) ;
int min_cmp = MIN ( a_len , b_len ) ;
for ( int i = 0 ; i < min_cmp ; i + + ) {
if ( operator [ ] ( i ) < p_array [ i ] ) {
return true ;
} else if ( p_array [ i ] < operator [ ] ( i ) ) {
return false ;
}
}
return a_len < b_len ;
}
bool Array : : operator < = ( const Array & p_array ) const {
return ! operator > ( p_array ) ;
}
bool Array : : operator > ( const Array & p_array ) const {
return p_array < * this ;
}
bool Array : : operator > = ( const Array & p_array ) const {
return ! operator < ( p_array ) ;
}
2014-02-10 02:10:30 +01:00
uint32_t Array : : hash ( ) const {
2017-03-05 16:44:50 +01:00
uint32_t h = hash_djb2_one_32 ( 0 ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < _p - > array . size ( ) ; i + + ) {
h = hash_djb2_one_32 ( _p - > array [ i ] . hash ( ) , h ) ;
2014-02-10 02:10:30 +01:00
}
return h ;
}
2020-04-21 00:06:00 +02:00
void Array : : _assign ( const Array & p_array ) {
if ( _p - > typed . type ! = Variant : : OBJECT & & _p - > typed . type = = p_array . _p - > typed . type ) {
//same type or untyped, just reference, shuold be fine
_ref ( p_array ) ;
} else if ( _p - > typed . type = = Variant : : NIL ) { //from typed to untyped, must copy, but this is cheap anyway
_p - > array = p_array . _p - > array ;
} else if ( p_array . _p - > typed . type = = Variant : : NIL ) { //from untyped to typed, must try to check if they are all valid
2020-04-21 17:16:45 +02:00
if ( _p - > typed . type = = Variant : : OBJECT ) {
//for objects, it needs full validation, either can be converted or fail
for ( int i = 0 ; i < p_array . _p - > array . size ( ) ; i + + ) {
if ( ! _p - > typed . validate ( p_array . _p - > array [ i ] , " assign " ) ) {
return ;
}
2020-04-21 00:06:00 +02:00
}
2020-04-21 17:16:45 +02:00
_p - > array = p_array . _p - > array ; //then just copy, which is cheap anyway
} else {
//for non objects, we need to check if there is a valid conversion, which needs to happen one by one, so this is the worst case.
Vector < Variant > new_array ;
new_array . resize ( p_array . _p - > array . size ( ) ) ;
for ( int i = 0 ; i < p_array . _p - > array . size ( ) ; i + + ) {
Variant src_val = p_array . _p - > array [ i ] ;
if ( src_val . get_type ( ) = = _p - > typed . type ) {
new_array . write [ i ] = src_val ;
} else if ( Variant : : can_convert_strict ( src_val . get_type ( ) , _p - > typed . type ) ) {
Variant * ptr = & src_val ;
Callable : : CallError ce ;
2020-11-09 04:19:09 +01:00
Variant : : construct ( _p - > typed . type , new_array . write [ i ] , ( const Variant * * ) & ptr , 1 , ce ) ;
2020-04-21 17:16:45 +02:00
if ( ce . error ! = Callable : : CallError : : CALL_OK ) {
ERR_FAIL_MSG ( " Unable to convert array index " + itos ( i ) + " from ' " + Variant : : get_type_name ( src_val . get_type ( ) ) + " ' to ' " + Variant : : get_type_name ( _p - > typed . type ) + " '. " ) ;
}
} else {
ERR_FAIL_MSG ( " Unable to convert array index " + itos ( i ) + " from ' " + Variant : : get_type_name ( src_val . get_type ( ) ) + " ' to ' " + Variant : : get_type_name ( _p - > typed . type ) + " '. " ) ;
}
}
_p - > array = new_array ;
2020-04-21 00:06:00 +02:00
}
} else if ( _p - > typed . can_reference ( p_array . _p - > typed ) ) { //same type or compatible
_ref ( p_array ) ;
} else {
ERR_FAIL_MSG ( " Assignment of arrays of incompatible types. " ) ;
}
}
void Array : : operator = ( const Array & p_array ) {
_assign ( p_array ) ;
2014-02-10 02:10:30 +01:00
}
2020-05-14 14:29:06 +02:00
2017-03-05 16:44:50 +01:00
void Array : : push_back ( const Variant & p_value ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND ( ! _p - > typed . validate ( p_value , " push_back " ) ) ;
2014-02-10 02:10:30 +01:00
_p - > array . push_back ( p_value ) ;
}
2020-11-08 21:09:45 +01:00
void Array : : append_array ( const Array & p_array ) {
ERR_FAIL_COND ( ! _p - > typed . validate ( p_array , " append_array " ) ) ;
_p - > array . append_array ( p_array . _p - > array ) ;
}
2014-02-10 02:10:30 +01:00
Error Array : : resize ( int p_new_size ) {
return _p - > array . resize ( p_new_size ) ;
}
2017-03-05 16:44:50 +01:00
void Array : : insert ( int p_pos , const Variant & p_value ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND ( ! _p - > typed . validate ( p_value , " insert " ) ) ;
2017-03-05 16:44:50 +01:00
_p - > array . insert ( p_pos , p_value ) ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
void Array : : erase ( const Variant & p_value ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND ( ! _p - > typed . validate ( p_value , " erase " ) ) ;
2014-02-10 02:10:30 +01:00
_p - > array . erase ( p_value ) ;
}
2016-11-18 21:30:16 +01:00
Variant Array : : front ( ) const {
2019-08-15 04:57:49 +02:00
ERR_FAIL_COND_V_MSG ( _p - > array . size ( ) = = 0 , Variant ( ) , " Can't take value from empty array. " ) ;
2016-11-18 21:30:16 +01:00
return operator [ ] ( 0 ) ;
}
Variant Array : : back ( ) const {
2019-08-15 04:57:49 +02:00
ERR_FAIL_COND_V_MSG ( _p - > array . size ( ) = = 0 , Variant ( ) , " Can't take value from empty array. " ) ;
2016-11-18 21:30:16 +01:00
return operator [ ] ( _p - > array . size ( ) - 1 ) ;
}
2017-03-05 16:44:50 +01:00
int Array : : find ( const Variant & p_value , int p_from ) const {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " find " ) , - 1 ) ;
2016-06-10 19:57:56 +02:00
return _p - > array . find ( p_value , p_from ) ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
int Array : : rfind ( const Variant & p_value , int p_from ) const {
2020-05-14 16:41:43 +02:00
if ( _p - > array . size ( ) = = 0 ) {
2016-06-03 23:10:43 +02:00
return - 1 ;
2020-05-14 16:41:43 +02:00
}
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " rfind " ) , - 1 ) ;
2016-06-03 23:10:43 +02:00
2016-06-10 22:28:09 +02:00
if ( p_from < 0 ) {
// Relative offset from the end
p_from = _p - > array . size ( ) + p_from ;
}
if ( p_from < 0 | | p_from > = _p - > array . size ( ) ) {
// Limit to array boundaries
p_from = _p - > array . size ( ) - 1 ;
}
2017-03-05 16:44:50 +01:00
for ( int i = p_from ; i > = 0 ; i - - ) {
if ( _p - > array [ i ] = = p_value ) {
2016-06-03 23:10:43 +02:00
return i ;
2019-07-01 12:59:42 +02:00
}
}
2016-06-03 23:10:43 +02:00
return - 1 ;
}
2017-03-05 16:44:50 +01:00
int Array : : find_last ( const Variant & p_value ) const {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " find_last " ) , - 1 ) ;
2016-06-10 22:28:09 +02:00
return rfind ( p_value ) ;
}
2017-03-05 16:44:50 +01:00
int Array : : count ( const Variant & p_value ) const {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " count " ) , 0 ) ;
2020-05-14 16:41:43 +02:00
if ( _p - > array . size ( ) = = 0 ) {
2016-06-03 23:10:43 +02:00
return 0 ;
2020-05-14 16:41:43 +02:00
}
2016-06-03 23:10:43 +02:00
2017-03-05 16:44:50 +01:00
int amount = 0 ;
for ( int i = 0 ; i < _p - > array . size ( ) ; i + + ) {
if ( _p - > array [ i ] = = p_value ) {
2016-06-03 23:10:43 +02:00
amount + + ;
2019-07-01 12:59:42 +02:00
}
}
2016-06-03 23:10:43 +02:00
return amount ;
}
2017-03-05 16:44:50 +01:00
bool Array : : has ( const Variant & p_value ) const {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " use 'has' " ) , false ) ;
2016-07-02 19:03:35 +02:00
return _p - > array . find ( p_value , 0 ) ! = - 1 ;
}
2014-02-10 02:10:30 +01:00
void Array : : remove ( int p_pos ) {
_p - > array . remove ( p_pos ) ;
}
2017-03-05 16:44:50 +01:00
void Array : : set ( int p_idx , const Variant & p_value ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND ( ! _p - > typed . validate ( p_value , " set " ) ) ;
2017-03-05 16:44:50 +01:00
operator [ ] ( p_idx ) = p_value ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
const Variant & Array : : get ( int p_idx ) const {
2014-02-10 02:10:30 +01:00
return operator [ ] ( p_idx ) ;
}
2018-03-09 20:16:08 +01:00
Array Array : : duplicate ( bool p_deep ) const {
2017-09-06 23:13:05 +02:00
Array new_arr ;
int element_count = size ( ) ;
new_arr . resize ( element_count ) ;
2020-04-21 00:06:00 +02:00
new_arr . _p - > typed = _p - > typed ;
2017-09-06 23:13:05 +02:00
for ( int i = 0 ; i < element_count ; i + + ) {
2018-03-09 20:16:08 +01:00
new_arr [ i ] = p_deep ? get ( i ) . duplicate ( p_deep ) : get ( i ) ;
2017-09-06 23:13:05 +02:00
}
return new_arr ;
}
2019-07-17 03:31:58 +02:00
2020-05-01 18:06:36 +02:00
int Array : : _clamp_slice_index ( int p_index ) const {
int arr_size = size ( ) ;
int fixed_index = CLAMP ( p_index , - arr_size , arr_size - 1 ) ;
if ( fixed_index < 0 ) {
fixed_index = arr_size + fixed_index ;
}
return fixed_index ;
2019-07-17 03:31:58 +02:00
}
Array Array : : slice ( int p_begin , int p_end , int p_step , bool p_deep ) const { // like python, but inclusive on upper bound
2019-09-24 21:33:09 +02:00
2020-05-01 18:06:36 +02:00
Array new_arr ;
2019-07-17 03:31:58 +02:00
2020-05-01 18:06:36 +02:00
ERR_FAIL_COND_V_MSG ( p_step = = 0 , new_arr , " Array slice step size cannot be zero. " ) ;
2019-07-17 03:31:58 +02:00
2020-12-15 13:04:21 +01:00
if ( is_empty ( ) ) { // Don't try to slice empty arrays.
2019-07-17 03:31:58 +02:00
return new_arr ;
2020-05-14 16:41:43 +02:00
}
2020-05-01 18:06:36 +02:00
if ( p_step > 0 ) {
2020-05-14 16:41:43 +02:00
if ( p_begin > = size ( ) | | p_end < - size ( ) ) {
2020-05-01 18:06:36 +02:00
return new_arr ;
2020-05-14 16:41:43 +02:00
}
2020-05-01 18:06:36 +02:00
} else { // p_step < 0
2020-05-14 16:41:43 +02:00
if ( p_begin < - size ( ) | | p_end > = size ( ) ) {
2019-07-17 03:31:58 +02:00
return new_arr ;
2020-05-14 16:41:43 +02:00
}
2019-07-17 03:31:58 +02:00
}
2020-05-01 18:06:36 +02:00
int begin = _clamp_slice_index ( p_begin ) ;
int end = _clamp_slice_index ( p_end ) ;
int new_arr_size = MAX ( ( ( end - begin + p_step ) / p_step ) , 0 ) ;
new_arr . resize ( new_arr_size ) ;
if ( p_step > 0 ) {
int dest_idx = 0 ;
for ( int idx = begin ; idx < = end ; idx + = p_step ) {
ERR_FAIL_COND_V_MSG ( dest_idx < 0 | | dest_idx > = new_arr_size , Array ( ) , " Bug in Array slice() " ) ;
new_arr [ dest_idx + + ] = p_deep ? get ( idx ) . duplicate ( p_deep ) : get ( idx ) ;
2019-07-17 03:31:58 +02:00
}
2020-05-01 18:06:36 +02:00
} else { // p_step < 0
int dest_idx = 0 ;
for ( int idx = begin ; idx > = end ; idx + = p_step ) {
ERR_FAIL_COND_V_MSG ( dest_idx < 0 | | dest_idx > = new_arr_size , Array ( ) , " Bug in Array slice() " ) ;
new_arr [ dest_idx + + ] = p_deep ? get ( idx ) . duplicate ( p_deep ) : get ( idx ) ;
2019-07-17 03:31:58 +02:00
}
}
return new_arr ;
}
2014-02-10 02:10:30 +01:00
struct _ArrayVariantSort {
2017-03-05 16:44:50 +01:00
_FORCE_INLINE_ bool operator ( ) ( const Variant & p_l , const Variant & p_r ) const {
bool valid = false ;
2014-02-10 02:10:30 +01:00
Variant res ;
2017-03-05 16:44:50 +01:00
Variant : : evaluate ( Variant : : OP_LESS , p_l , p_r , res , valid ) ;
2020-05-14 16:41:43 +02:00
if ( ! valid ) {
2017-03-05 16:44:50 +01:00
res = false ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
return res ;
}
} ;
2020-10-13 20:59:37 +02:00
void Array : : sort ( ) {
2014-02-10 02:10:30 +01:00
_p - > array . sort_custom < _ArrayVariantSort > ( ) ;
}
struct _ArrayVariantSortCustom {
2020-11-23 17:38:46 +01:00
Object * obj = nullptr ;
2014-02-10 02:10:30 +01:00
StringName func ;
2017-03-05 16:44:50 +01:00
_FORCE_INLINE_ bool operator ( ) ( const Variant & p_l , const Variant & p_r ) const {
const Variant * args [ 2 ] = { & p_l , & p_r } ;
2020-02-19 20:27:19 +01:00
Callable : : CallError err ;
2017-03-05 16:44:50 +01:00
bool res = obj - > call ( func , args , 2 , err ) ;
2020-05-14 16:41:43 +02:00
if ( err . error ! = Callable : : CallError : : CALL_OK ) {
2017-03-05 16:44:50 +01:00
res = false ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
return res ;
}
} ;
2020-10-13 20:59:37 +02:00
void Array : : sort_custom ( Object * p_obj , const StringName & p_function ) {
ERR_FAIL_NULL ( p_obj ) ;
2014-02-10 02:10:30 +01:00
2018-01-09 19:03:51 +01:00
SortArray < Variant , _ArrayVariantSortCustom , true > avs ;
2017-03-05 16:44:50 +01:00
avs . compare . obj = p_obj ;
avs . compare . func = p_function ;
2017-11-25 04:07:54 +01:00
avs . sort ( _p - > array . ptrw ( ) , _p - > array . size ( ) ) ;
2014-02-10 02:10:30 +01:00
}
2018-01-10 19:36:53 +01:00
void Array : : shuffle ( ) {
const int n = _p - > array . size ( ) ;
2020-05-14 16:41:43 +02:00
if ( n < 2 ) {
2018-01-10 19:36:53 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2018-01-10 19:36:53 +01:00
Variant * data = _p - > array . ptrw ( ) ;
for ( int i = n - 1 ; i > = 1 ; i - - ) {
const int j = Math : : rand ( ) % ( i + 1 ) ;
const Variant tmp = data [ j ] ;
data [ j ] = data [ i ] ;
data [ i ] = tmp ;
}
}
2017-11-02 19:04:38 +01:00
template < typename Less >
_FORCE_INLINE_ int bisect ( const Vector < Variant > & p_array , const Variant & p_value , bool p_before , const Less & p_less ) {
int lo = 0 ;
int hi = p_array . size ( ) ;
if ( p_before ) {
while ( lo < hi ) {
const int mid = ( lo + hi ) / 2 ;
if ( p_less ( p_array . get ( mid ) , p_value ) ) {
lo = mid + 1 ;
} else {
hi = mid ;
}
}
} else {
while ( lo < hi ) {
const int mid = ( lo + hi ) / 2 ;
if ( p_less ( p_value , p_array . get ( mid ) ) ) {
hi = mid ;
} else {
lo = mid + 1 ;
}
}
}
return lo ;
}
int Array : : bsearch ( const Variant & p_value , bool p_before ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " binary search " ) , - 1 ) ;
2017-11-02 19:04:38 +01:00
return bisect ( _p - > array , p_value , p_before , _ArrayVariantSort ( ) ) ;
}
int Array : : bsearch_custom ( const Variant & p_value , Object * p_obj , const StringName & p_function , bool p_before ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND_V ( ! _p - > typed . validate ( p_value , " custom binary search " ) , - 1 ) ;
2017-11-02 19:04:38 +01:00
ERR_FAIL_NULL_V ( p_obj , 0 ) ;
_ArrayVariantSortCustom less ;
less . obj = p_obj ;
less . func = p_function ;
return bisect ( _p - > array , p_value , p_before , less ) ;
}
2020-10-13 20:59:37 +02:00
void Array : : invert ( ) {
2014-02-10 02:10:30 +01:00
_p - > array . invert ( ) ;
}
2017-03-05 16:44:50 +01:00
void Array : : push_front ( const Variant & p_value ) {
2020-04-21 00:06:00 +02:00
ERR_FAIL_COND ( ! _p - > typed . validate ( p_value , " push_front " ) ) ;
2017-03-05 16:44:50 +01:00
_p - > array . insert ( 0 , p_value ) ;
2015-12-12 12:27:30 +01:00
}
2017-03-05 16:44:50 +01:00
Variant Array : : pop_back ( ) {
2020-12-15 13:04:21 +01:00
if ( ! _p - > array . is_empty ( ) ) {
2016-08-27 17:33:45 +02:00
int n = _p - > array . size ( ) - 1 ;
Variant ret = _p - > array . get ( n ) ;
_p - > array . resize ( n ) ;
return ret ;
}
return Variant ( ) ;
2015-12-12 12:27:30 +01:00
}
2017-03-05 16:44:50 +01:00
Variant Array : : pop_front ( ) {
2020-12-15 13:04:21 +01:00
if ( ! _p - > array . is_empty ( ) ) {
2016-08-27 17:33:45 +02:00
Variant ret = _p - > array . get ( 0 ) ;
2015-12-12 12:27:30 +01:00
_p - > array . remove ( 0 ) ;
2016-08-27 17:33:45 +02:00
return ret ;
}
return Variant ( ) ;
2015-12-12 12:27:30 +01:00
}
2018-08-23 20:31:02 +02:00
Variant Array : : min ( ) const {
Variant minval ;
for ( int i = 0 ; i < size ( ) ; i + + ) {
if ( i = = 0 ) {
minval = get ( i ) ;
} else {
bool valid ;
Variant ret ;
Variant test = get ( i ) ;
Variant : : evaluate ( Variant : : OP_LESS , test , minval , ret , valid ) ;
if ( ! valid ) {
return Variant ( ) ; //not a valid comparison
}
if ( bool ( ret ) ) {
//is less
minval = test ;
}
}
}
return minval ;
}
Variant Array : : max ( ) const {
Variant maxval ;
for ( int i = 0 ; i < size ( ) ; i + + ) {
if ( i = = 0 ) {
maxval = get ( i ) ;
} else {
bool valid ;
Variant ret ;
Variant test = get ( i ) ;
Variant : : evaluate ( Variant : : OP_GREATER , test , maxval , ret , valid ) ;
if ( ! valid ) {
return Variant ( ) ; //not a valid comparison
}
if ( bool ( ret ) ) {
//is less
maxval = test ;
}
}
}
return maxval ;
}
2019-04-20 01:57:29 +02:00
const void * Array : : id ( ) const {
return _p - > array . ptr ( ) ;
}
2020-04-21 00:06:00 +02:00
Array : : Array ( const Array & p_from , uint32_t p_type , const StringName & p_class_name , const Variant & p_script ) {
_p = memnew ( ArrayPrivate ) ;
_p - > refcount . init ( ) ;
set_typed ( p_type , p_class_name , p_script ) ;
_assign ( p_from ) ;
}
void Array : : set_typed ( uint32_t p_type , const StringName & p_class_name , const Variant & p_script ) {
ERR_FAIL_COND_MSG ( _p - > array . size ( ) > 0 , " Type can only be set when array is empty. " ) ;
ERR_FAIL_COND_MSG ( _p - > refcount . get ( ) > 1 , " Type can only be set when array has no more than one user. " ) ;
ERR_FAIL_COND_MSG ( _p - > typed . type ! = Variant : : NIL , " Type can only be set once. " ) ;
ERR_FAIL_COND_MSG ( p_class_name ! = StringName ( ) & & p_type ! = Variant : : OBJECT , " Class names can only be set for type OBJECT " ) ;
Ref < Script > script = p_script ;
ERR_FAIL_COND_MSG ( script . is_valid ( ) & & p_class_name = = StringName ( ) , " Script class can only be set together with base class name " ) ;
_p - > typed . type = Variant : : Type ( p_type ) ;
_p - > typed . class_name = p_class_name ;
_p - > typed . script = script ;
_p - > typed . where = " TypedArray " ;
}
2017-03-05 16:44:50 +01:00
Array : : Array ( const Array & p_from ) {
2020-04-02 01:20:12 +02:00
_p = nullptr ;
2014-02-10 02:10:30 +01:00
_ref ( p_from ) ;
}
2018-08-23 20:31:02 +02:00
2017-01-11 12:53:31 +01:00
Array : : Array ( ) {
2017-03-05 16:44:50 +01:00
_p = memnew ( ArrayPrivate ) ;
2014-02-10 02:10:30 +01:00
_p - > refcount . init ( ) ;
}
2020-05-14 14:29:06 +02:00
2014-02-10 02:10:30 +01:00
Array : : ~ Array ( ) {
_unref ( ) ;
}