2021-03-03 13:59:55 -08:00
< ? php declare ( strict_types = 1 );
/*
* Copyright ( c ) 2020 https :// rewiv . com sikofitt @ gmail . com
*
* This file is a part of Olive BBS
*
* This Source Code Form is subject to the
* terms of the Mozilla Public License , v . 2.0 .
*
* If a copy of the MPL was not distributed with this file ,
* You can obtain one at https :// mozilla . org / MPL / 2.0 /.
*
* ___ ___ ___
* ( ) .-. ( ) ( )
* .--. | | ( __ ) ___ ___ .--. | |.-. | |.-. .--.
* / \ | | ( '' " ( )( / \ | / \ | / \ / _ \
* | .-. | | | | | | | | .-. | .-. | .-. |. . ' ` . ;
* | | | | | | | | | | | | | | | | | | | || ' | |
* | | | | | | | | | | | |/ | | | | | | | _\_ ` . ( ___ )
* | | | | | | | | | | | ' _.| | | | | | ( ). ' .
* | ' | | | | | ' ' ; | .' .-| ' | | ' | || | ` \ |
* ' `-' | | | | \ `' /' ` - ' ' `-' ;' ` - ' ; ; ' . _ , ' '
* `.__.(___(___) '_.' ` . __ . ' `.__. `.__. ' . ___ . '
*
*/
namespace Olivebbs\Map ;
use ArrayAccess ;
2022-05-11 13:17:06 -07:00
use function class_exists ;
2021-03-03 13:59:55 -08:00
use Countable ;
use Ds\Map ;
2022-05-11 13:17:06 -07:00
use function enum_exists ;
2021-03-03 13:59:55 -08:00
use function is_array ;
2021-03-04 11:16:57 -08:00
use function is_callable ;
2021-03-03 13:59:55 -08:00
use function is_int ;
use function is_object ;
use function is_string ;
2022-05-11 13:17:06 -07:00
use function mb_strlen ;
2022-05-11 16:28:23 -07:00
use Olivebbs\Map\Enum\KeyType ;
use Olivebbs\Map\Enum\ValueType ;
2021-03-03 13:59:55 -08:00
use Olivebbs\Map\Exception\InvalidArgumentException ;
2022-05-11 13:17:06 -07:00
use function sprintf ;
2021-03-03 13:59:55 -08:00
class GenericMap implements ArrayAccess , Countable
{
protected Map $map ;
2022-05-11 16:28:23 -07:00
private readonly string $valueTypeValue ;
private readonly string $keyTypeValue ;
2021-03-03 13:59:55 -08:00
2022-05-11 16:28:23 -07:00
public function __construct ( protected KeyType $keyType = KeyType :: ANY , protected object | string $valueType = ValueType :: ANY )
{
$this -> keyTypeValue = $this -> keyType -> value ;
$this -> valueTypeValue = is_string ( value : $this -> valueType ) ? $this -> valueType : $this -> valueType -> value ;
2021-03-03 13:59:55 -08:00
2022-05-11 13:17:06 -07:00
if ( ! $this -> isValidValueType ( type : $this -> valueType )) {
2022-05-11 16:28:23 -07:00
throw new InvalidArgumentException ( message : sprintf ( 'Invalid value type (%s)' , $this -> valueTypeValue ));
2021-03-03 13:59:55 -08:00
}
$this -> map = new Map ();
}
2022-05-11 16:28:23 -07:00
public function getKeyType () : KeyType
2021-03-03 13:59:55 -08:00
{
return $this -> keyType ;
}
2022-05-11 16:28:23 -07:00
public function getValueType () : object | string
2021-03-03 13:59:55 -08:00
{
return $this -> valueType ;
}
/**
* @ inheritDoc
*/
2022-05-11 13:17:06 -07:00
public function offsetExists ( mixed $offset ) : bool
2021-03-03 13:59:55 -08:00
{
2022-05-11 13:17:06 -07:00
$this -> checkTypes ( offset : $offset );
2021-03-03 13:59:55 -08:00
2022-05-11 13:17:06 -07:00
return $this -> map -> offsetExists ( offset : $offset );
2021-03-03 13:59:55 -08:00
}
/**
* @ inheritDoc
*/
2022-05-11 13:17:06 -07:00
public function offsetGet ( mixed $offset ) : mixed
2021-03-03 13:59:55 -08:00
{
2022-05-11 13:17:06 -07:00
$this -> checkTypes ( offset : $offset );
2021-03-03 13:59:55 -08:00
2022-05-11 13:17:06 -07:00
if ( ! $this -> map -> offsetExists ( offset : $offset )) {
2021-03-03 13:59:55 -08:00
return null ;
}
2022-05-11 13:17:06 -07:00
return $this -> map -> offsetGet ( offset : $offset );
2021-03-03 13:59:55 -08:00
}
/**
* @ inheritDoc
*/
2022-05-11 13:17:06 -07:00
public function offsetSet ( mixed $offset , mixed $value ) : void
2021-03-03 13:59:55 -08:00
{
2022-05-11 16:28:23 -07:00
$this -> checkTypes ( offset : $offset , value : $value );
2021-03-03 13:59:55 -08:00
2022-05-11 13:17:06 -07:00
$this -> map -> offsetSet ( offset : $offset , value : $value );
2021-03-03 13:59:55 -08:00
}
/**
* @ inheritDoc
*/
2022-05-11 13:17:06 -07:00
public function offsetUnset ( mixed $offset ) : void
2021-03-03 13:59:55 -08:00
{
2022-05-11 13:17:06 -07:00
$this -> checkTypes ( offset : $offset );
2021-03-03 13:59:55 -08:00
2022-05-11 13:17:06 -07:00
$this -> map -> offsetUnset ( offset : $offset );
2021-03-03 13:59:55 -08:00
}
/**
* Count elements of an object
*
* @ link https :// php . net / manual / en / countable . count . php
* @ return int The custom count as an integer .
* </ p >
* < p >
* The return value is cast to an integer .
*/
public function count () : int
{
return $this -> map -> count ();
}
public function toArray () : array
{
return $this -> map -> toArray ();
}
protected function assertInitialValues ( array $initialValues ) : bool
{
foreach ( $initialValues as $key => $value ) {
2022-05-11 16:28:23 -07:00
if ( ! $this -> checkType ( $this -> keyType , $key ) || ! $this -> checkType ( $this -> valueType , $value )) {
throw new InvalidArgumentException ( message : sprintf ( 'Invalid types for map [%s => %s], they should be [%s => %s]' , get_debug_type ( value : $key ), get_debug_type ( value : $value ), $this -> keyType -> value , is_string ( $this -> valueType ) ? $this -> valueType : $this -> valueType -> value ));
2021-03-03 13:59:55 -08:00
}
}
return true ;
}
2022-05-11 13:17:06 -07:00
private function checkTypes ( mixed $offset , mixed $value = null ) : void
{
if ( is_array ( value : $offset ) || is_object ( value : $offset )) {
throw new InvalidArgumentException ( message : 'Map keys cannot be objects or arrays' );
}
if ( ! $this -> checkType ( type : $this -> keyType , var : $offset )) {
2022-05-11 16:28:23 -07:00
throw new InvalidArgumentException ( message : sprintf ( 'Key should be of value %s.' , $this -> keyTypeValue ));
2022-05-11 13:17:06 -07:00
}
if ( null !== $value && ! $this -> checkType ( type : $this -> valueType , var : $value )) {
2022-05-11 16:28:23 -07:00
throw new InvalidArgumentException ( message : sprintf ( 'Value should be of type %s.' , $this -> valueTypeValue ));
2022-05-11 13:17:06 -07:00
}
}
2021-03-03 13:59:55 -08:00
/**
* @ codeCoverageIgnore
2022-05-11 16:28:23 -07:00
* @ param KeyType | ValueType | string $type
* @ param mixed $var
2021-03-03 13:59:55 -08:00
*
* @ return bool
*/
2022-05-11 16:28:23 -07:00
private function checkType ( KeyType | ValueType | string $type , mixed $var ) : bool
2021-03-03 13:59:55 -08:00
{
2022-05-11 13:17:06 -07:00
return match ( $type ) {
2022-05-11 16:28:23 -07:00
ValueType :: OBJECT => is_object ( value : $var ),
ValueType :: CALLABLE => is_callable ( value : $var ),
ValueType :: ARRAY => is_array ( value : $var ),
KeyType :: INT , KeyType :: INTEGER , ValueType :: INT , ValueType :: INTEGER => is_int ( value : $var ),
KeyType :: STRING , ValueType :: STRING => is_string ( value : $var ),
KeyType :: CHAR , ValueType :: CHAR => is_string ( value : $var ) && ( function_exists ( 'mb_strlen' ) ? mb_strlen ( string : $var ) === 1 : strlen ( string : $var ) === 1 ),
ValueType :: ENUM => is_object ( value : $var ) && enum_exists ( enum : $var :: class ),
KeyType :: ANY , ValueType :: ANY => true ,
2022-05-11 13:17:06 -07:00
default => $var instanceof $type ,
};
2021-03-03 13:59:55 -08:00
}
2022-05-11 13:17:06 -07:00
2022-05-11 16:28:23 -07:00
private function isValidValueType ( object | string $type ) : bool
2021-03-03 13:59:55 -08:00
{
2022-05-11 16:28:23 -07:00
if ( is_string ( value : $type ) && class_exists ( class : $type )) {
2021-03-04 10:51:09 -08:00
return true ;
}
2022-05-11 16:28:23 -07:00
return $type instanceof ValueType ;
2021-03-03 13:59:55 -08:00
}
}