PHP Carbon class changing my original variable value

PhpLaravelPhp Carbon

Php Problem Overview


I'm trying to make a few navigation buttons in a calendar type thing I'm creating, and I'm using carbon to create the dates.

This is the code in the controller:

if ($date == null) {
	$date = \Carbon\Carbon::now();
} else {
	$date = \Carbon\Carbon::createFromFormat('Y-m-d', $date);
}
$navDays = [
	'-7Days' => $date->subDay('7')->toDateString(),
	'-1Day'  => $date->subDay('1')->toDateString(),
	'Today'    => $date->today()->toDateString(),
	'+1Day'  => $date->addDay('1')->toDateString(),
	'+7Days' => $date->addDay('7')->toDateString()
];

and then I'm my view, I'm doing this:

@foreach($navDays as $key => $i)
	<li>
		<a href="/planner/bookings/{{ $i }}" class="small button">
			{{ $key }}
		</a>
	</li>
@endforeach

This problem is, that carbon seems to change the $date during the array creating, because these are the dates I'm getting(with $date being set to 2015-11-29):

<ul class="button-group even-5">
	<li><a href="/planner/bookings/2015-11-22" class="small button">-7Days</a></li>
	<li><a href="/planner/bookings/2015-11-21" class="small button">-1Day</a></li>
	<li><a href="/planner/bookings/2015-12-22" class="small button">Today</a></li>
	<li><a href="/planner/bookings/2015-11-22" class="small button">+1Day</a></li>
	<li><a href="/planner/bookings/2015-11-29" class="small button">+7Days</a></li>
</ul>

Does anybody know what I'm doing wrong?

Php Solutions


Solution 1 - Php

When you run these methods against a Carbon object it updates the object itself. Therefore addDay() moves the value of Carbon one day forward.

Here's what you need to do:

$now = Carbon::now();

$now->copy()->addDay();
$now->copy()->addMonth();
$now->copy()->addYear();
// etc...

The copy method essentially creates a new Carbon object which you can then apply the changes to without affecting the original $now variable.

To sum up, the methods for copying a Carbon instance are:

  • copy
  • clone - an alias of copy

Check out the documentation: https://carbon.nesbot.com/docs/

Solution 2 - Php

The problem is that you're assuming that subDay()/addDay() don't change the date object, whereas they do.... they're just wrapping around the DateTime object modify() method:

> DateTime::modify -- date_modify — Alters the timestamp

(my emphasis)

Instead, use

$navDays = [
    '-7Days' => (clone $date)->subDay('7')->toDateString(),
    '-1Day'  => (clone $date)->subDay('1')->toDateString(),
    'Today'  => (clone $date)->today()->toDateString(),
    '+1Day'  => (clone $date)->addDay('1')->toDateString(),
    '+7Days' => (clone $date)->addDay('7')->toDateString()
];

Solution 3 - Php

Doco says

> You can also create a copy() of an existing Carbon instance. As expected the date, time and timezone values are all copied to the new instance.

$dt = Carbon::now();
echo $dt->diffInYears($dt->copy()->addYear());  // 1

// $dt was unchanged and still holds the value of Carbon:now()

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionJohan Bj&#246;rklundView Question on Stackoverflow
Solution 1 - PhpdiggersworldView Answer on Stackoverflow
Solution 2 - PhpMark BakerView Answer on Stackoverflow
Solution 3 - PhpYevgeniy AfanasyevView Answer on Stackoverflow