IMUZA.com

Xserver<<WordPress(ConoHa)<<はてなブログ

ホーム / javascript / はてなブログカードの iframe を div に差し替える(自サイトのみ)

はてなブログカードの iframe を div に差し替える(自サイトのみ)

2019/03/19 javascript, はてなブログ

長々とこだわっている「自サイトブログカードのリンクターゲット」の件、そろそろ実装できるものに仕上げないといけません(笑)。

これまでこんなことをやってきました。

  • はてなブログカードを DIV要素で自作する(自サイトのみ)
  • Node.js でブログカードを返すアプリを作ってみた
  • はてなブログカードの iframe 内を弄る(自サイトのみ)

  • hatenablog-parts.com/embed を 自サイト/embed に
  • XMLHttpRequest を使い embed-wrapper を取り出す
  • CSS

hatenablog-parts.com/embed を 自サイト/embed に

前回は、XMLHttpRequest を使ってターゲットファイルを読み込み、そこからOGPデータを取り出してブログカードを作りましたが、ふと、自サイト名/embed でもブログカードはできることを思い出し、であれば、同じく XMLHttpRequest で 自サイト/embed/エントリー にアクセスすればブログカードそのもののファイルが取れるのではないかと思いつきました。

そうすれば、記事作成時に特別にマークアップする必要もなく、通常の iframe のままにしておき、それを div に置き換えることもできそうです。

で、やってみたのが次のスクリプトです。

XMLHttpRequest を使い embed-wrapper を取り出す

「非同期処理を Promise と then で書き直す」をご覧ください。

(function(){
    var article = document.getElementsByTagName('article')[0];
    if(article !== undefined){
        var iframes = article.getElementsByTagName('iframe');
        var reg = new RegExp('http(s)?%3A%2F%2F' + location.hostname + '.+$');
        Array.prototype.forEach.call(iframes, function(iframe){
            var embedUrl = reg.exec(iframe.src);
            if(embedUrl !== null){
                iframe.src = '';
                var url = decodeURIComponent(embedUrl[0]).replace('entry', 'embed');


                var xhr = new XMLHttpRequest();
                var parser = new DOMParser();


                xhr.open('GET', url);
                xhr.send();


                xhr.onreadystatechange = function() {
                    if(xhr.readyState === 4 && xhr.status === 200) {
                        var doc = parser.parseFromString(xhr.response, "text/html");
                        var blogcard = doc.getElementsByClassName('embed-wrapper')[0];
                        var element = blogcard.getElementsByTagName('time')[0];
                        element.removeAttribute('data-relative');
                        var links = blogcard.getElementsByTagName('a');
                        Array.prototype.forEach.call(links, function(link){
                            link.setAttribute('target', '_top');
                        });
                        iframe.parentNode.replaceChild(blogcard, iframe);
                    }
                }
            }
        });
    }
})();

  • ブログカードは記事内にしかありえませんから article タグに限定し、
  • 複数あるかも知れない iframe を取り出し、
  • ひとつひとつの iframe の src が自サイトであるかをチェックし、
  • URL の entry を embed に差し替えて HTML ファイルを読み込み、
  • DOM パースして <div class="embed-wrapper"></div> だけ取り出し、
  • リンクターゲットを _top に差し替え、
  • iframe と div を入れ替え

ています。

一番上が他サイト(当サイトの記事ですがテスト用サイトから呼んでいる)、2つめから下が自サイトのブログカードです。リンクをクリックすると同じタブで開きます。

CSS

.embed-wrapper {
font-family: -apple-system,BlinkMacSystemFont,Segoe UI,Helvetica,Arial,sans-serif;
max-width: 500px;
background-color: #fff;
font-size: 13px;
border: 1px solid rgba(0, 0, 0, 0.1);
border-radius: 3px;
background-clip: padding-box;
box-sizing: border-box; }
.embed-wrapper * {
word-wrap: break-word;
word-break: break-all; }
.embed-wrapper-inner {
padding: 12px; }
.embed-header {
position: relative;
padding-left: 35px;
height: 30px;
margin-bottom: 5px; }
.embed-header .blog-icon {
position: absolute;
top: 0;
left: 0;
width: 30px;
height: 30px;
border-radius: 30px;
background-clip: padding-box; }
.embed-header .blog-title,
.embed-header .embed-header .blog-user {
font-size: 12px;
line-height: 1.3;
overflow: hidden;
max-height: 15px; }
.embed-header .blog-title a,
.embed-header .embed-header .blog-user a {
color: #0e6ace; }
.embed-header .blog-title {
margin: 0;
color: #0e6ace;
margin-right: 110px; }
.embed-header .blog-user {
margin-right: 110px; }
.embed-header .blog-user a {
color: #6b7377; }
.embed-logo {
position: absolute;
top: 0;
right: 0; }
.embed-logo .embed-logo-img {
vertical-align: middle;
opacity: .6;
width: 100px;
height: 18px; }
.embed-content {
position: relative;
height: 100px;
overflow: hidden; }
.embed-content.with-thumb {
min-height: 100px; }
.embed-content.with-thumb .entry-body {
margin-right: 110px; }
.embed-content .thumb-wrapper {
position: absolute;
top: 0;
right: 0; }
.embed-content .thumb-wrapper .thumb {
width: 100px;
height: 100px; }
.embed-content .entry-title {
font-family: inherit;
font-size: 17px;
margin: 0 0 2px;
line-height: 1.3;
max-height: 47px;
overflow: hidden; }
.embed-content .entry-content {
line-height: 1.5;
font-size: 12px;
max-height: 72px;
overflow: hidden;
display: -webkit-box;
-webkit-box-orient: vertical;
-webkit-line-clamp: 3; }
.embed-footer {
margin-top: 8px;
height: 15px;
position: relative;
font-size: 11px;
color: #6b7377; }
.embed-footer a {
color: #6b7377; }
.embed-footer .updated {
margin-right: .3em; }
.embed-footer .star-count {
padding-bottom: 1px; }
.embed-footer img {
vertical-align: middle; }

こだわってオリジナルのものとまったく同じにしてみました。

よし、これを実装しよう!

Joomla! 3.9.4 セキュリティ & バグフィックスがリリースされています
非同期処理を Promise と then で書き直す
Twitter
Facebook
ブックマーク
LINEで送る

最初のサイドバー

最新記事

2023/03/21

WordPress:メール送信にGmailのSMTPを使う(プラグインなし)

2023/03/10

WordPress:サムネイル付き次の記事/前の記事

2023/02/28

Twitterシェアリンクはintent/tweetを使おう

2023/02/21

GoogleタグマネージャーとアナリティクスGA4

2023/02/8

WordPress:人気記事を直近の期間指定で表示する

最新記事を一覧で見る

よく読まれている記事

よく読まれている記事を一覧で見る

カテゴリー

  • はてなブログ214
  • WebTips109
  • javascript98
  • Joomla!88
  • Windows68
  • Wordpress65
  • CSS63
  • Joomla!更新53
  • Linux49
  • はてなテーマ45
  • Google34
  • Plamo33
  • はてなプラグイン25
  • php22
  • Node.js18
  • Ubuntu16
  • SASS16
  • laravel415
  • Chrome11
  • cms-style10
  • iPhone9
  • ConoHa WING6
  • Git入門6
  • genesis6
  • Python5
  • Android5
  • PC全般4
  • Facebook4
  • スマートフォン4
  • 静的サイトジェネレーター3
  • Firefox3
  • SSD3
  • Blankslate3
  • Docker3
  • Twitter2
  • GitHub2
  • Mactype2
  • はてなブクマ1
  • 映画1
  • youtube1
  • rails入門1
  • Xserver1

Footer

My Web Sites

  • @半径とことこ60分
  • そんなには褒めないよ。映画評
  • IMUZA.com
  • GitHub

Related Sites

  • WordPress公式
  • WordPress関数リファレンス
  • PHPマニュアル

Contact Us

  • お問い合わせフォーム
  • Twitter
  • Facebook
  • Feedly

Copyright © 2023 · IMUZA.com