CakePHP 3 の Html ヘルパーを使って data:image 形式の画像を表示する


CakePHP 3.6 の Html ヘルパーを使って data:image 形式の画像を表示してみました。

課題

画像がファイルとして存在していなく、Base64エンコーディングされているデータだったり、1回のリクエストでHTMLと画像を送信しなければならないときなど、

<img src="...S3wF==">

というタグをヘルパーを使って出力しようとして、テンプレート内で、

<?= $this->Html->image('...S3wF=='); ?>

とそのまま data:image を突っ込んでも、

<img src="app...S3wF==" alt="">

の "app" のようなアプリケーションのパスがついてしまってうまく表示してくれません。

解決法

Htmlヘルパーの image() メソッドは、内部的にはパスを整形して formatTemplate() メソッドを呼び出しているようなので、

<?php
$imageData = '...S3wF==';
echo $this->Html->formatTemplate(
  'image', [ 'url' => $imageData ]
);
?>

で良さそうです。

<img src="...S3wF==" alt="">

src 以外の属性を付けたい場合は、

<?php
$imageData = '...S3wF==';
$attributes = [
    'class' => 'header-image header-image-logo',
    'alt' => 'Header Image'
];
echo $this->Html->formatTemplate(
    'image', [
      'url' => $imageData, 
      'attrs' => $this->Html->templater()->formatAttributes($attributes)
    ]
);
?>

とすると、

<img src="...S3wF==" class="header-image header-image-logo" alt="Header Image"/>

のような結果が得られました。

ICOOON MONOさんから、無料のタブレットアイコン素材2 の素材を使わせていただいて、実際に表示される画像をスニペット的に作りました。
ビューにコピペしていただければ動くと思います。
なお、ヒアドキュメントを使う場合、trim() と併用して前後のスペースを除くようにしておくのが無難ですね。

<?php

$imageData = trim(<<<END_OF_IMAGE

END_OF_IMAGE
);

$attributes = [
    'class' => 'icon-tablet',
    'alt' => 'Tablet'
];

echo $this->Html->formatTemplate(
    'image', [
        'url' => trim($imageData),
        'attrs' => $this->Html->templater()->formatAttributes($attributes)
    ]
);

?>