<?php /** * This file is part of the Carbon package. * * (c) Brian Nesbitt <brian@nesbot.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Carbon\Traits; use Closure; use Generator; use ReflectionClass; use ReflectionException; use ReflectionMethod; use Throwable; /** * Trait Mixin. * * Allows mixing in entire classes with multiple macros. */ trait Mixin { /** * Stack of macro instance contexts. * * @var array */ protected static $macroContextStack = []; /** * Mix another object into the class. * * @example * ``` * Carbon::mixin(new class { * public function addMoon() { * return function () { * return $this->addDays(30); * }; * } * public function subMoon() { * return function () { * return $this->subDays(30); * }; * } * }); * $fullMoon = Carbon::create('2018-12-22'); * $nextFullMoon = $fullMoon->addMoon(); * $blackMoon = Carbon::create('2019-01-06'); * $previousBlackMoon = $blackMoon->subMoon(); * echo "$nextFullMoon\n"; * echo "$previousBlackMoon\n"; * ``` * * @param object|string $mixin * * @throws ReflectionException * * @return void */ public static function mixin($mixin) { \is_string($mixin) && trait_exists($mixin) ? self::loadMixinTrait($mixin) : self::loadMixinClass($mixin); } /** * @param object|string $mixin * * @throws ReflectionException */ private static function loadMixinClass($mixin) { $methods = (new ReflectionClass($mixin))->getMethods( ReflectionMethod::IS_PUBLIC | ReflectionMethod::IS_PROTECTED ); foreach ($methods as $method) { if ($method->isConstructor() || $method->isDestructor()) { continue; } $method->setAccessible(true); static::macro($method->name, $method->invoke($mixin)); } } /** * @param string $trait */ private static function loadMixinTrait($trait) { $context = eval(self::getAnonymousClassCodeForTrait($trait)); $className = \get_class($context); foreach (self::getMixableMethods($context) as $name) { $closureBase = Closure::fromCallable([$context, $name]); static::macro($name, function () use ($closureBase, $className) { /** @phpstan-ignore-next-line */ $context = isset($this) ? $this->cast($className) : new $className(); try { // @ is required to handle error if not converted into exceptions $closure = @$closureBase->bindTo($context); } catch (Throwable $throwable) { // @codeCoverageIgnore $closure = $closureBase; // @codeCoverageIgnore } // in case of errors not converted into exceptions $closure = $closure ?: $closureBase; return $closure(...\func_get_args()); }); } } private static function getAnonymousClassCodeForTrait(string $trait) { return 'return new class() extends '.static::class.' {use '.$trait.';};'; } private static function getMixableMethods(self $context): Generator { foreach (get_class_methods($context) as $name) { if (method_exists(static::class, $name)) { continue; } yield $name; } } /** * Stack a Carbon context from inside calls of self::this() and execute a given action. * * @param static|null $context * @param callable $callable * * @throws Throwable * * @return mixed */ protected static function bindMacroContext($context, callable $callable) { static::$macroContextStack[] = $context; try { return $callable(); } finally { array_pop(static::$macroContextStack); } } /** * Return the current context from inside a macro callee or a null if static. * * @return static|null */ protected static function context() { return end(static::$macroContextStack) ?: null; } /** * Return the current context from inside a macro callee or a new one if static. * * @return static */ protected static function this() { return end(static::$macroContextStack) ?: new static(); } }
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
Boundaries.php | File | 11.04 KB | 0644 |
|
Cast.php | File | 1.06 KB | 0644 |
|
Comparison.php | File | 34.63 KB | 0644 |
|
Converter.php | File | 14.84 KB | 0644 |
|
Creator.php | File | 30.37 KB | 0644 |
|
Date.php | File | 161.43 KB | 0644 |
|
DeprecatedProperties.php | File | 1.57 KB | 0644 |
|
Difference.php | File | 52.91 KB | 0644 |
|
IntervalRounding.php | File | 1.54 KB | 0644 |
|
IntervalStep.php | File | 2.29 KB | 0644 |
|
Localization.php | File | 28.59 KB | 0644 |
|
Macro.php | File | 3.02 KB | 0644 |
|
MagicParameter.php | File | 721 B | 0644 |
|
Mixin.php | File | 4.58 KB | 0644 |
|
Modifiers.php | File | 13.44 KB | 0644 |
|
Mutability.php | File | 1.3 KB | 0644 |
|
ObjectInitialisation.php | File | 422 B | 0644 |
|
Options.php | File | 12.78 KB | 0644 |
|
Rounding.php | File | 7.54 KB | 0644 |
|
Serialization.php | File | 8.29 KB | 0644 |
|
Test.php | File | 7.56 KB | 0644 |
|
Timestamp.php | File | 6.44 KB | 0644 |
|
ToStringFormat.php | File | 1.43 KB | 0644 |
|
Units.php | File | 11.89 KB | 0644 |
|
Week.php | File | 7.22 KB | 0644 |
|