Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Symfony2 - Array to string conversion exception with a flash message

I set a flash message in a controller with the following code:

$this->get('session')->getFlashBag()->add('success', 'Message sent successfully');

And in my template, I use the following to (attempt to) display it:

{% if app.session.flashbag.has('success') %}
    <div id="flash">
        {{ app.session.flashbag.get('success') }}
    </div>
{% endif %}

The problem is that, despite the API documentation stating that get returns a string, I'm getting an array to string conversion exception. If I change the code in the template to:

{% for flashMessage in app.session.flashbag.get('success') %}
    <div id="flash">
        {{ flashMessage }}
    </div>
{% endfor %}

It works perfectly. I'd rather not use a loop here since I'm only ever either going to have the single message or not.

Is there a solution where I can just check for the existence of a single flash message and display it if it's there? Or am I stuck with a useless loop?

like image 401
Major Productions Avatar asked Mar 23 '13 01:03

Major Productions


3 Answers

Solved it by indexing at 0:

{{ app.session.flashbag.get('success')[0] }}

My suspicions were correct - get returns an array rather than a string. Here's the flashbag's add method:

public function add($type, $message)
{
    $this->flashes[$type][] = $message;
}

And get:

public function get($type, array $default = array())
{
    if (!$this->has($type)) {
        return $default;
    }

    $return = $this->flashes[$type];

    unset($this->flashes[$type]);

    return $return;
}

They need to fix the API documentation so it reflects reality. They should also provide an elegant way to handle a single flash message.

EDIT: A backwards compatible (PHP 5.3 and below) version -

{% if app.session.flashbag.has('success') %}
    {% set flashbag = app.session.flashbag.get('success') %}
    {% set message = flashbag[0] %}
    <div id="flash">
        {{ message }}
    </div>
{% endif %}
like image 190
Major Productions Avatar answered Nov 12 '22 09:11

Major Productions


For one flash message:

{{ app.session.flashbag.get('success')[0] }}

For all:

{% for type, messages in app.session.flashbag.all() %}
    {% for message in messages %}
        <div class="alert alert-{{ type }}">
            {{ message }}
        </div>
    {% endfor %}
{% endfor %}
like image 37
mkjasinski Avatar answered Nov 12 '22 09:11

mkjasinski


I've just hit this myself. It was because I was using the add() method instead of set().

The differences between Add and Set:

public function add($type, $message)
{
    $this->flashes[$type][] = $message;
}

The above would add an extra array which isn't required in this case.

Whereas:

public function set($type, $messages)
{
    $this->flashes[$type] = (array) $messages;
}

So set() results in $array[$key] = $value, rather than what add does, which is $array[$key][] = $value which is what is causing your Array to string conversion because you're passing an array, not a string.

like image 45
Matt Cavanagh Avatar answered Nov 12 '22 10:11

Matt Cavanagh