0
votes

I am trying to setup a weighted tag cloud, that works when tags are a string property in the blog entity.

Now I've setup tags as it's own entity and related to blog as a bi-directional ManyToMany/ManyToMany relationship.

Needless to say it's not working, I'm guessing it's because tag is now it's own object.

My question: What am I doing wrong here now that tags is it's own entity and not a string property from blog entity?

When I dump $tagWeights from the controller below, I get an error when I should be seeing something like this:

array (size=78)

'Tag1' => float 1

'Tag2' => float 5

'Tag3' => float 2

error: (this line:

foreach ($tags as $tag) {

$tagWeights[$tag] = (isset($tagWeights[$tag])) ? $tagWeights[$tag] + 1 : 1;

}

ContextErrorException: Warning: Illegal offset type in isset or empty in /var/www/html/Satori/src/Symfony/AcmeBundle/Entity/TagRepository.php line 34

I'm calling tags in the following way in twig:

Twig

{% for tag, weight in tags %}
    <span class="weight-{{ weight }}"><a href="">{{ tag }}</a></span>
{% else %}
    <p>There are no tags</p>
{% endfor %}

Controller (dumping $tags works great shows all the tags)

public function footerAction()
{
    $em = $this->getDoctrine()->getManager();

    $tags = $em->getRepository('AcmeBundle:Tag')
        ->getTags();

    $tagWeights = $em->getRepository('AcmeBundle:Tag')
        ->getTagWeights($tags);
    var_dump($tagWeights); die();
    return array(
        'tags' => $tagWeights,

    );
}

Here is the getTags and getTagWeights:

public function getTags()
{
    $tags = $this->createQueryBuilder('t')
        ->select('t.tag')
        ->getQuery()
        ->getResult();

    return $tags;
}

public function getTagWeights($tags)
{
    $tagWeights = array();

    if (empty($tags))
        return $tagWeights;

    foreach ($tags as $tag)
    {
        $tagWeights[$tag] = (isset($tagWeights[$tag['tag']])) ? $tagWeights[$tag] + 1 : 1;
    }

    // Shuffle the tags
    uksort($tagWeights, function() {
        return rand() > rand();
    });

    $max = max($tagWeights);

    // Max of 5 weights
    $multiplier = ($max > 5) ? 5 / $max : 1;

    foreach ($tagWeights as &$tag)
    {
        $tag = ceil($tag * $multiplier);
    }

    return $tagWeights;
}

Tag entity

class Tag
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @var string
     *
     * @ORM\Column(name="tag", type="string", length=255)
     */
    private $tag;

    /**
     * @ORM\ManyToMany(targetEntity="Blog", mappedBy="tags")
     */
    protected $blogs;

    public function __construct()
    {
        $this->blogs = new ArrayCollection();
    }

    /**
     * Get id
     *
     * @return integer 
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * Set tag
     *
     * @param string $tag
     * @return Tag
     */
    public function setTag($tag)
    {
        $this->tag = $tag;

        return $this;
    }

    /**
     * Get tag
     *
     * @return string 
     */
    public function getTag()
    {
        return $this->tag;
    }

    /**
     * Add blogs
     *
     * @param \AcmeBundle\Entity\Blog $blogs
     * @return Tag
     */
    public function addBlog(\AcmeBundle\Entity\Blog $blogs)
    {
        $this->blogs[] = $blogs;

        return $this;
    }

    /**
     * Remove blogs
     *
     * @param \AcmeBundle\Entity\Blog $blogs
     */
    public function removeBlog(\AcmeBundle\Entity\Blog $blogs)
    {
        $this->blogs->removeElement($blogs);
    }

    /**
     * Get blogs
     *
     * @return \Doctrine\Common\Collections\Collection 
     */
    public function getBlogs()
    {
        return $this->blogs;
    }
}

Proposed solutions and results

Using (isset($tagWeights[$tag['tag']])) error:

ContextErrorException: Warning: Illegal offset type in /var/www/html/Satori/src/Symfony/AcmeBundle/Entity/TagRepository.php line 34

Using (isset($tagWeights[$tag[$tag->getTag()])) error:

ContextErrorException: Notice: Trying to get property of non-object in /var/www/html/Satori/src/Symfony/AcmeBundle/Entity/TagRepository.php line 34

1
I haven't read through your entire code, but does it have anything to do with the fact that your declaration for getTagWeights() does not include the $tags argument?Thomas Kelley
Ah yes, fixed that now I get nice fat error: ContextErrorException: Warning: Illegal offset type in isset or empty in /var/www/html/Satori/src/Symfony/AcmeBundle/Entity/TagRepository.php line 34 (this line: $tagWeights[$tag] = (isset($tagWeights[$tag])) ? $tagWeights[$tag] + 1 : 1;) Will update question.dizzyd

1 Answers

0
votes

I think I know what's happening here...

When you make this call in your footerAction:

$tags = $em->getRepository('AcmeBundle:Tag')
    ->getTags();

your repository returns to you an array of arrays. So when you say:

isset($tagWeights[$tag])

...that $tag is actually an associative array of the result set. I think what you mean to say here is:

isset($tagWeights[$tag['tag']])

That said, I'd advise against giving your Tag entity a property called tag. a tag doesn't have a tag, it has a name, or a description. And it'd make much more sense here to say

isset($tagWeights[$tag['name']])

...or by having the repository return the Tag entities themselves (why do you even need to override the built-in getXyz() method?), and do:

isset($tagWeights[$tag->getName()])