ポンコツエンジニアのごじゃっぺ開発日記。

いろいろポンコツだけど、気にするな。エンジニアの日々の開発などの記録を残していきます。 自動で収入を得られるサービスやシステムを作ることが目標!!

【無料】簡単に独自アイキャッチ画像を生成できる仕組みを作る

まずは、今回の独自システムで作成したアイキャッチ画像はこちらになります。

f:id:ponkotsu0605:20210603101111p:plain

今回は、これを作成できる仕組みを作ったので、それを紹介しようと思います。

概要

完全無料で作りたかったので、有料のサーバは用意していません。そのため、簡単に無料でスクリプトを実行できるGoogle Apps Script(以下、GAS)を利用しました。

ウェブサイトの公開をGASで行ってはいるのですが、HTML(とJava Scriptによるスクリプト)だけなので、HTMLファイルをPC上に作成するだけでも利用することができます。 自分の場合は、PCが変わっても利用したかったので、GASを利用してウェブサイトに公開しています。

ざっくりとしたフローは以下になります。

  1. ブックマーク等からGASで作成したウェブサイトを開く
  2. 記事タイトルを入力する(window.promptの利用)
  3. 自動でcanvasにゴリゴリ作成して画像を作成
  4. 生成された画像をダウンロード

HTMLのソースコード

早速ですが、今回プログラミングしたソースコードを紹介したいと思います。

とりあえず動けばいいやって思って作成したので、雑なコードになっていますが。。

<!DOCTYPE html>
<html>
    <head>
       <base target="_top">
       <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.6.0/jquery.min.js"></script>
       <script src="https://ajax.googleapis.com/ajax/libs/webfont/1.6.26/webfont.js"></script>
       <link href="https://fonts.googleapis.com/css2?family=Kosugi+Maru&display=swap" rel="stylesheet">
   </head>
    <body>
        <p style="font-family: 'Kosugi Maru', sans-serif;">ポンコツエンジニアのごじゃっぺ開発日記。</p>
        <canvas id="canvas" width="1200" height="630"></canvas>
        <script>
           var text =   window.prompt("記事タイトルを入力してください", "");
           var fontSize = 68;
           // フォントを利用するためにいったんHTMLに出力する
           $('p').text(text);
           // canvas内で画像を使用するために、一度セットしておく
           img = new Image();
           img.src = "";
            $(function(){
                setTimeout(function() {
                    var canvas = document.getElementById('canvas');
                    if (canvas.getContext) {
                        var context = canvas.getContext('2d');
                        context.fillStyle = "rgb(251, 176, 59)";
                        context.fillRect(0,0,1200,630);

                        // 影
                        context.shadowColor = "rgb(0, 0, 0)";
                        context.shadowOffsetX = 3;
                        context.shadowOffsetY = 3;
                        context.shadowBlur = 3;

                        context.fillStyle = "rgb(255, 255, 255)";
                        var diff = 40;
                        var r = 20;
                        context.beginPath();
                        context.moveTo(diff,630 / 2);

                        context.arcTo(diff,diff, 1200 - diff,diff, r);
                        context.arcTo(1200 - diff, diff, 1200 - diff, 630 - diff, r);
                        context.arcTo(1200 - diff,630 - diff, diff,630 - diff, r);
                        context.arcTo(diff,630 - diff, diff,diff, r);
                        context.lineTo(diff,diff);
                        context.closePath();
                        context.fill()

                        context.fillStyle = "rgb(0, 0, 0)";

                        context.shadowOffsetX = 0;
                        context.shadowOffsetY = 0;
                        context.shadowBlur = 0;
                        context.font = "bold " + fontSize + "px 'Kosugi Maru', sans-serif";

                        printAt(context, text, 100, 180, fontSize * 1.5, 1000)

                        fontSize = 30
                        context.font = "bold " + fontSize + "px 'Kosugi Maru', sans-serif";
                        printAt(context, 'ポンコツエンジニアのごじゃっぺ開発日記。', 230, 520, fontSize * 1.5, 1000)

                        context.beginPath();
                        context.arc(100 + 50, 450 + 50, 50, 0, 2 * Math.PI);
                        context.clip();

                        context.beginPath();
                        context.drawImage(img, 0, 0, 100, 100, 100, 450, 100, 100);
                        let link = document.createElement('a');
                        link.href = canvas.toDataURL('image/png');
                        link.download = text + '.png';
                        link.click();

                        window.close();
                    }
                }, 1000); // フォントダウンロード時間のため遅らせて描画させる
            })

            // https://www.it-swarm-ja.com/ja/javascript/html5%E3%82%AD%E3%83%A3%E3%83%B3%E3%83%90%E3%82%B9ctxfilltext%E3%81%AF%E6%94%B9%E8%A1%8C%E3%82%92%E3%81%97%E3%81%BE%E3%81%9B%E3%82%93%E3%81%8B%EF%BC%9F/971773981/
            function printAt( context , text, x, y, lineHeight, fitWidth)
            {
                fitWidth = fitWidth || 0;
                if (fitWidth <= 0)
                {
                    context.fillText( text, x, y );
                    return;
                }
                for (var idx = 1; idx <= text.length; idx++)
                {
                    var str = text.substr(0, idx);
                    if (context.measureText(str).width > fitWidth)
                    {
                        context.fillText( text.substr(0, idx-1), x, y );
                        printAt(context, text.substr(idx-1), x, y + lineHeight, lineHeight, fitWidth);
                        return;
                    }
                }
                context.fillText( text, x, y );
            }
        </script>
    </body>
</html>

ぜひ、JavaScriptが書ける方は読んでいただきたいのですが、少し工夫した点を紹介したいと思います。

Webフォントを利用する工夫

Webフォントを利用するために、Google Fontsを利用しています。

fonts.google.com

canvasでWebフォントを表示するための工夫として、HTML上で一度表示したい文字列をそのフォントで表示してからではないと、canvas内で出力できなかったので、あえてpタグで文字を出力させています。

また、記事タイトルの入力後にも、jQueryを利用してpタグの文字を置き換えてからcanvasで出力するようにしています。

ダウンロードで時間がかかるっぽいので、canvasに描画するのはsetTimeout()を利用してあえて1秒遅延させてからにしています。

独自の画像を表示させるための工夫

canvas内に自分のアイコンを表示させています。

f:id:ponkotsu0605:20210603102507p:plain

しかしGAS上で画像を表示するためには、同じドメインからアクセスできるように、同じところに画像をアップロードしておく必要があります。(CORS:クロスオリジンの対策)

ただ、GAS上だと画像のアップロードができなかったので、表示させたい画像をbase64にエンコードさせて、その文字列をHTML上にベタ書きしました。ソースコードは汚いですが、実現できればOKなのです。

画像をbase64エンコードした文字列に変換させるためには、以下のサイトを利用させていただきました。

lab.syncer.jp

canvasでの文字列の自動改行

記事タイトルが画像の右端まで行ったら改行したいです。 そこで、以下のサイトのソースコードが良さそうだったので、丸コピしてきました。

www.it-swarm-ja.com

canvas内の描画のがんばり

以下のcanvasリファレンスを頑張って読んで勉強しました。久しぶりなので、canvasのいろいろを忘れてました。

www.htmq.com

HTMLをウェブサイトとして公開

上記で作成したHTMLを後は公開するだけです。

GASのソースコードをウェブサイトとして公開する方法は、以下の記事にまとめているので、そちらを参考にしてみてください。

www.pnkts.net

公開したウェブサイトをブックマーク

上記で公開したウェブサイトのURLをブックマークしておけば、簡単に呼び出すことができます。

さいごに

今回、このようにして、簡単にアイキャッチ画像を作成するシステムを構築しました。

以降、このブログのアイキャッチ画像に適用していけたらなと思います。(記事内画像がないものに関して)

お問い合わせプライバシーポリシー制作物