Applying automatic link class to images embedded to posts

I’d really love to be able to conditionally apply a CSS class to link elements of images embedded into posts, but I can’t seem to figure this one out.

Basically what I’d like to achieve is to replace the default embedded image link that links the image to the full size version of itself to opening it in a colorbox.

Read More

I know I can achieve this with a bit of Javascript-trickery, but I’d love to figure out how to do this in the server side and then just attach the colorbox functionality in a straightforward manner.

So, is there a filter I should add or what’s the route to take here. Ideally I’d love to be able to write the code so that if a user embeds an image, the link opens the full size version in a colorbox but if the user specifies a custom link, it’ll work like a regular link. This shouldn’t be that hard.

Thanks a lot for any help in advance!

Related posts

Leave a Reply

4 comments

  1. Thanks a LOT to user t31os for the tip on where to find my solution!

    Also thanks to user orionrush for pointing out a dumb mistake I had made! 🙂

    Here’s how I got what I wanted:

    function add_colorbox_class_to_image_links($html, $attachment_id, $attachment) {
        $linkptrn = "/<a[^>]*>/";
        $found = preg_match($linkptrn, $html, $a_elem);
    
        // If no link, do nothing
        if($found <= 0) return $html;
    
        $a_elem = $a_elem[0];
    
        // Check to see if the link is to an uploaded image
        $is_attachment_link = strstr($a_elem, "wp-content/uploads/");
    
        // If link is to external resource, do nothing
        if($is_attachment_link === FALSE) return $html;
    
        if(strstr($a_elem, "class="") !== FALSE){ // If link already has class defined inject it to attribute
            $a_elem_new = str_replace("class="", "class="colorbox ", $a_elem);
            $html = str_replace($a_elem, $a_elem_new, $html);
        }else{ // If no class defined, just add class attribute
            $html = str_replace("<a ", "<a class="colorbox" ", $html);
        }
    
        return $html;
    }
    
    add_filter('image_send_to_editor', 'add_colorbox_class_to_image_links', 10, 3);
    
  2. (Posting this answer because this question shows up on Google and I couldn’t find what I wanted through Google.)

    Brief Explanation and Example

    After retrieving the post from the data base, this will put all of the specified classes into the anchor tag whenever there is an anchor tag with only an image tag inside of it.

    It works with many image tags in one post, as well as a variety of other weird possibilities. For instance, something like this

    <article>
        <a href="an_image.jpg">
            <img src="an_image.jpg">
        </a>
        <a class="media-img" href="another_image.jpg">
            <img src="another_image.jpg">
        </a>
        <p>Text with a <a href="google.com">link</a></p>
        <a class="big gray ugly" href="third_image.jpg">
            <img src="third_image.jpg">
        </a>
        <a foo="bar" class="big" href="fourth_image.jpg">
            <img src="fourth_image.jpg">
        </a>
    </article>
    

    will become

    <article>
        <a class="media-img" href="an_image.jpg">
            <img src="an_image.jpg">
        </a>
        <a class="media-img media-img" href="another_image.jpg">
            <img src="another_image.jpg">
        </a>
        <p>Text with a <a href="google.com">link</a></p>
        <a class="media-img big gray ugly" href="third_image.jpg">
            <img src="third_image.jpg">
        </a>
        <a foo="bar" class="media-img big" href="fourth_image.jpg">
            <img src="fourth_image.jpg">
        </a>
    </article>
    

    Code (for functions.php)

    function add_classes_to_linked_images($html) {
        $classes = 'media-img'; // can do multiple classes, separate with space
    
        $patterns = array();
        $replacements = array();
    
        $patterns[0] = '/<a(?![^>]*class)([^>]*)>s*<img([^>]*)>s*</a>/'; // matches img tag wrapped in anchor tag where anchor tag where anchor has no existing classes
        $replacements[0] = '<a1 class="' . $classes . '"><img3></a>';
    
        $patterns[1] = '/<a([^>]*)class="([^"]*)"([^>]*)>s*<img([^>]*)>s*</a>/'; // matches img tag wrapped in anchor tag where anchor has existing classes contained in double quotes
        $replacements[1] = '<a1class="' . $classes . ' 2"3><img4></a>';
    
        $patterns[2] = '/<a([^>]*)class='([^']*)'([^>]*)>s*<img([^>]*)>s*</a>/'; // matches img tag wrapped in anchor tag where anchor has existing classes contained in single quotes
        $replacements[2] = '<a1class="' . $classes . ' 2"3><img4></a>';
    
        $html = preg_replace($patterns, $replacements, $html);
    
        return $html;
    }
    
    add_filter('the_content', 'add_classes_to_linked_images', 100, 1);
    

    Other Notes

    • In the first regular expression pattern, (?![^>]*class) is a negative lookahead so that the first regex replace rule only affects <a href...><img></a>, not <a class... href...><img></a>. (Read more on lookarounds.)
    • In regular expressions for this, I think [^>]* is better than .*. [^>]* means zero or more characters that are not a >. Without [^>]*, I think there could be problems if there are multiple > characters on one line or in other weird situations.
    • In the regular expressions, the backslash followed by a number in the replacements such as in '<a1 class="' . $classes . '"><img3></a>' refers to the stuff inside corresponding parenthetical block in the corresponding pattern. In other words, 1 means “put the stuff that matches what’s inside the first set of parentheses”.
    • In the line add_filter('the_content', 'add_classes_to_linked_images', 100, 1);, the first parameter is the filter for getting the content of a post from the database, the second parameter is the name of the function we want to use, the third parameter is the priority of the filter (higher numbers get executed later), and the fourth parameter is the number of arguments for the filter.
    • Assuming your anchor and image combination already has the class you want to add, this function will cause it to show up twice in the source code of your page. (Don’t worry, it will only show up twice at most and it won’t cause any issues. See example above.)
  3. Someone correct me if im wrong here, but the above gives me warnings on $astart – I believe these lines:

       if(strstr($astart, "class="") !== FALSE){ // If link already has class defined inject it to attribute
        $a_elem_new = str_replace("class="", "class="colorbox ", $astart);
    

    should be:

    if(strstr($a_elem, "class="") !== FALSE){ // If link already has class defined inject it to attribute
        $a_elem_new = str_replace("class="", "class="colorbox ", $a_elem);
    

    but I may not be reading it correctly. . .

    also we could dynamically add the uploads folder rather then hardcode it:

        $uploads = wp_upload_dir();
    $upload_path = basename($uploads['baseurl']); // dosen't account for multi site - not sure how to do that. . .
    $is_attachment_link = strstr($a_elem, $upload_path);
    

    could probably substitute wp_basename but not sure of the advantages…