banner
叶星优酸乳

叶星优酸乳

阅读是砍向内心冰封大海的斧头
twitter
tg_channel
mastodon

Obsidian|共有コンテンツの迅速なカード生成

宣言:本チュートリアルは、元々 Telegram Obsidian グループの友人 @稲米鼠(ブログ記事)から派生したもので、後に開発者の友人 @海龍 によって最適化され、最終的にこの記事のバージョンとなりました。

image

以前の投稿で、共有コンテンツを迅速にカードに変換する方法を共有しましたが、今回はその完全なプロセスを共有します。

前提条件#

  • Templater:コミュニティマーケットの Templater プラグインをインストールして有効化
  • テンプレートファイル(tp - 生成文字カード)を指定のフォルダに保存:ダウンロードテンプレート例をクリックするか、下の付録からコピー
  • スクリプト js ファイル(get_tweet_card)を指定のフォルダに保存:ダウンロード js ファイル完全コードをクリックするか、下の付録からコピー

注:

  1. Templater プラグイン内でテンプレートフォルダ、スクリプトフォルダを設定し、それぞれ Template folder location、Script files folder location に配置します。
  2. 異なるテンプレートを設定し、異なるアバターやニックネームを構成して、異なるカードを生成できます。

操作手順#

上記の準備が整ったら、楽しく使用できます。具体的な手順は以下の通りです:

  1. 設定-ショートカットキー で Templater テンプレートを呼び出すショートカットキーを設定します。私は ⌘+/ に設定しました。
  2. Obsidian の任意のファイル内で、テキストの一部を選択し、⌘+/ を押して、「tp - 生成文字カード」テンプレートを選択し、Enter を押すと、カードがクリップボードにコピーされます。

image


付録#

テンプレート例

<% tp.user.get_tweet_card(tp, {
  width: 1800,
  fontSize: 62,
  margin: 140,
  padding: 100,
  writeToClipboard: true,
  downloadToDisk: false,
  logo: `ここにカード内のあなたのアバターの base64 コードを置きます。例えば、以下のサイトで変換できます https://c.runoob.com/front-end/59/` ,
  name: 'あなたのニックネーム',
  userId: 'あなたの ID'
}) %>

js ファイルコード

/** @type {object} 設定項目 */
let opt = {}

/**
 * 初期設定
 *
 * @param {object} input
 */
const initOpt = (input, tp) => {
  opt = Object.assign({
    size: 'M',
    logo: AppLogo,
    appLogo: AppLogo,
    name: 'ここはユーザー名',
    userId: '@User_ID or anything',
    bgColors: ["#ffafbd", "#ffc3a0"],
    cardBgColor: 'rgba(255, 255, 255, .8)',
    contetnColor: '#333336',
    nameColor: '#333336',
    userIdColor: '#333336',
    timeColor: 'rgba(0, 0, 0, .5)',
    writeToClipboard: true,
    writeToDocument: false,
    downloadToDisk: false,
  }, input ? input : {})
  /** ==== 未設定の場合、デフォルト値を計算 ==== */
  /**
   * 属性が存在しない場合、デフォルト値を計算
   *
   * @param {*} key
   * @param {*} defVal
   */
  const setSubOpt = (key, defVal) => {
    if (!opt[key]) opt[key] = defVal
  }
  /** 画像の幅 */
  if (!opt.width) {
    switch (opt.size) {
      case 'S':
        opt.width = 480
        break;
      case 'M':
        opt.width = 700
        break;
      case 'L':
        opt.width = 960
        break;

      default:
        opt.width = 700
        break;
    }
  }
  /** 文字の大きさ */
  setSubOpt('fontSize', Math.round(opt.width / 30))
  setSubOpt('smallFontSize', Math.round(opt.fontSize * 0.6))
  /** 行の高さ */
  setSubOpt('lineHeight', 1.6)
  /** 段落の先頭のインデント */
  setSubOpt('indent', opt.fontSize * 2) /** 0に設定するとインデントなし */
  /** フォント */
  setSubOpt('fontFamily', 'Menlo, SFMono-Regular, Consolas, "Roboto Mono", "Source Code Pro", ui-sans-serif, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Inter", "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Microsoft YaHei", sans-serif')
  /** カードの外側の余白 */
  setSubOpt('margin', Math.round(opt.width / 15))
  setSubOpt('marginLR', opt.margin)
  setSubOpt('marginTB', opt.margin)
  /** カードの内側の余白 */
  setSubOpt('padding', Math.round(opt.width / 12))
  setSubOpt('paddingLR', opt.padding)
  setSubOpt('paddingTB', opt.padding)
  /** ロゴのサイズ */
  setSubOpt('logoSize', 2 * opt.fontSize)
  /** カードの角の丸み */
  setSubOpt('cardRadius', Math.round(opt.fontSize / 2))

  /** ==== 計算によって得られる値 ==== */

  opt.cardWidth = opt.width - opt.marginLR * 2
  opt.contentWidth = opt.cardWidth - opt.paddingLR * 2
  opt.contentMarginLR = opt.marginLR + opt.paddingLR
  opt.contentMarginTB = opt.marginTB + opt.paddingTB
  opt.paragraphsMarginBottom = Math.round(opt.fontSize / 2)
}

/**
 * 数字を2桁にする
 *
 * @param {number} num 0~99 の整数
 * @returnn {string}
 */
const dbNum = num => (num > 9 ? String(num) : '0' + num);
/** @type {array} */
const daysName = ['日', '月', '火', '水', '木', '金', '土']
/**
 * 現在の時間文字列を取得
 *
 * @return {string} 
 */
const getNowTime = () => {
  const now = new Date()
  const t = {
    YYYY: now.getFullYear(),
    MM: dbNum(now.getMonth() + 1),
    DD: dbNum(now.getDate()),
    hh: dbNum(now.getHours()),
    mm: dbNum(now.getMinutes()),
    ss: dbNum(now.getSeconds()),
    EE: daysName[now.getDay()]
  }
  return `${t.YYYY}-${t.MM}-${t.DD} ${t.EE} ${t.hh}:${t.mm}:${t.ss}`
}
// キャンバスオブジェクトを作成
const canvas = document.createElement('canvas')
const ctx = canvas.getContext('2d')

/**
 * キャンバスのテキストを行ごとに分割
 *
 * @param {object} ctx キャンバスのコンテキストオブジェクト
 * @param {string} text 書き込むテキスト内容
 * @param {number} width テキスト内容がキャンバスで占める幅
 * @return {array} 二次元配列、第一層は段落、第二層は段落内の各行
 */
const canvasTextSplit = (text, width) => {
  text = text.trim()
  if (text.length === 0) return []
  const result = []
  // まず段落を分割
  const paragraphArray = text.replace(/(\r?\n\s*)+/g, '\n').split(/\s*\r?\n\s*/g)
  for (const p of paragraphArray) {
    const linesInParagraph = []
    let nowLetter = 0
    for (let i = 0; i <= p.length; i++) {
      const thisLineWidth = linesInParagraph.length ? width : width - opt.indent
      if (ctx.measureText(p.substring(nowLetter, i)).width > thisLineWidth) {
        linesInParagraph.push(p.substring(nowLetter, i - 1))
        nowLetter = i - 1
      } else if (i === p.length) {
        linesInParagraph.push(p.substring(nowLetter, i))
      }
    }
    result.push(linesInParagraph)
  }
  return result
}
/**
 * 段落配列のテキストをキャンバスに描画
 *
 * @param {object} ctx キャンバスのコンテキストオブジェクト
 * @param {array} paragraphs 二次元配列、第一層は段落、第二層は段落内の各行
 * @param {number} startX 開始の横座標
 * @param {number} startY 開始の縦座標
 * @param {number} opt.lineHeight 行の高さ
 * @return {number} 終了位置の縦座標
 */
const drawText = async (paragraphs, startX, startY) => {
  let thisLineY = startY
  paragraphs.forEach((p, pIndex) => {
    p.forEach((line, lIndex) => {
      const thisLineX = lIndex ? startX : startX + opt.indent
      thisLineY += opt.lineHeight * opt.fontSize
      ctx.fillText(line, thisLineX, thisLineY)
    })
    thisLineY += opt.paragraphsMarginBottom
  })
  return thisLineY
}
/**
 * テキストを描画するのに必要な高さを計算
 *
 * @param {array} paragraphs 二次元配列、第一層は段落、第二層は段落内の各行
 * @param {number} opt.lineHeight 行の高さ
 * @return {number} テキスト内容が占める高さ
 */
const textNeedHeight = (paragraphs) => {
  return (paragraphs.length - 1) * opt.paragraphsMarginBottom
    + paragraphs.flat().length * opt.lineHeight * opt.fontSize
}
/**
 * base64 形式の画像を Blob 形式データに変換
 *
 * @param {string} dataUrl base64 形式のデータアドレス
 * @return {object} Blob 形式の画像データ
 */
const dataURLtoBlob = dataUrl => {
  const dataArr = dataUrl.split(',');
  const mime = dataArr[0].match(/:(.*?);/)[1];
  const bStr = atob(dataArr[1]);
  let n = bStr.length;
  const uint8Arr = new Uint8Array(n);
  while (n--) {
    uint8Arr[n] = bStr.charCodeAt(n);
  }
  return new Blob([uint8Arr], { type: mime });
}
/**
 * キャンバスを画像として保存し、自動的にダウンロード
 *
 * @param {object} canvas キャンバスオブジェクト
 * @param {string} name 保存するファイル名
 * @param {string} [type="png"] 画像ファイルの形式: png、jpeg、gif
 */
const downloadImgFromCanvas = (name) => {
  const imgDataUrl = canvas.toDataURL({ format: 'png', quality: 1 })
  const blob = dataURLtoBlob(imgDataUrl)
  const blobUrl = URL.createObjectURL(blob)
  const imgDownloadLink = document.createElement('a')
  imgDownloadLink.download = name + '.png'
  imgDownloadLink.href = blobUrl
  imgDownloadLink.click();
}

/**
 * 塗りつぶしの色を設定
 *
 * @param {string|array} colors
 */
const setFillColor = colors => {
  let fillColor
  if (typeof (colors) === 'string') {
    fillColor = colors
  } else if (colors.length === 1) {
    fillColor = colors[0]
  } else {
    fillColor = ctx.createLinearGradient(0, 0, opt.width, opt.width / 8);
    const pointStep = 1 / (colors.length - 1)
    colors.forEach((c, i) => {
      fillColor.addColorStop(i * pointStep, c);
    })
  }
  ctx.fillStyle = fillColor
}
/**
 * キャンバスのフォント設定
 *
 * @param {string|number} size
 * @param {string} color
 * @param {string} [weight='normal']
 * @param {string} [align='left']
 */
const setFont = (size, color, weight = 'normal', align = 'left') => {
  ctx.font = weight + ' ' + size + 'px ' + opt.fontFamily
  ctx.textAlign = align
  ctx.fillStyle = color
}
/**
 * キャンバスの影を設定
 *
 * @param {number} x
 * @param {number} y
 * @param {number} blur
 * @param {string} [color='rgba(0, 0, 0, 0)']
 */
const setShadow = (x, y, blur, color = 'rgba(0, 0, 0, 0)') => {
  ctx.shadowOffsetX = x
  ctx.shadowOffsetY = y
  ctx.shadowBlur = blur
  ctx.shadowColor = color
}
/**
 * キャンバスオブジェクトをリセット
 *
 * @param {number} height キャンバスの高さ
 * @param {string} fillColor キャンバスの背景色
 */
const canvasRest = height => {
  canvas.width = opt.width
  canvas.height = height
  setShadow(0, 0, 0)
  setFillColor(opt.bgColors)
  ctx.fillRect(0, 0, canvas.width, canvas.height)
}

/**
 * 角丸の四角形を描画
 *
 * @param {number} x
 * @param {number} y
 * @param {number} w
 * @param {number} h
 * @param {number} r
 */
const drawRoundedRect = (x, y, w, h, r) => {
  var ptA = { x: x + r, y: y }
  var ptB = { x: x + w, y: y }
  var ptC = { x: x + w, y: y + h }
  var ptD = { x: x, y: y + h }
  var ptE = { x: x, y: y }

  ctx.beginPath();

  ctx.moveTo(ptA.x, ptA.y);
  ctx.arcTo(ptB.x, ptB.y, ptC.x, ptC.y, r);
  ctx.arcTo(ptC.x, ptC.y, ptD.x, ptD.y, r);
  ctx.arcTo(ptD.x, ptD.y, ptE.x, ptE.y, r);
  ctx.arcTo(ptE.x, ptE.y, ptA.x, ptA.y, r);

  ctx.closePath()
  ctx.fill()
}

/**
 * 画像を同期的に読み込む
 *
 * @param {string} url
 * @param {number} l
 * @param {number} t
 */
const loadImage = async (url, l, t) => new Promise(resolve => {
  const img = new Image()
  img.onload = () => {
    ctx.drawImage(img, l, t, opt.logoSize, opt.logoSize)
    return resolve(true)
  }
  img.src = url
});

/**
 * 
 *
 * @param {*} tp
 * @return {*} 
 */
async function get_tweet_card(tp, input) {
  let selectedText = window.getSelection().toLocaleString() // 選択したテキストを取得

  /** @type {string} 入力を取得 */
  const inputContent = await tp.system.prompt('入力内容', selectedText, false, true)
  if (!inputContent) return selectedText

  /** 初期設定 */
  initOpt(input, tp)

  /** 内容を整理し、サイズを計算 */
  setFont(opt.fontSize, opt.contetnColor)
  const contentArr = canvasTextSplit(inputContent, opt.contentWidth)
  opt.contentHeight = textNeedHeight(contentArr)
  opt.cardHeight = opt.contentHeight
    + opt.paddingTB * 2
    + opt.logoSize
    + opt.lineHeight * opt.fontSize /** 時間を書くため */
    + 2 * opt.paragraphsMarginBottom /** 内容の上下に配置 */
  opt.height = opt.cardHeight + 2 * opt.marginTB
  /** キャンバスを初期化 */
  canvasRest(opt.height)
  /** カードを描画 */
  setShadow(0, 0, opt.margin * 0.6, 'rgba(0, 0, 0, .3)')
  ctx.fillStyle = opt.cardBgColor
  drawRoundedRect(opt.marginLR, opt.marginTB, opt.cardWidth, opt.cardHeight, opt.cardRadius)

  /** 内容のテキストを描画 */
  setFont(opt.fontSize, opt.contetnColor)
  setShadow(0, 0, 0)
  drawText(contentArr, opt.contentMarginLR, opt.contentMarginTB + opt.logoSize + opt.paragraphsMarginBottom)
  /** ユーザー名を描画 */
  setFont(opt.smallFontSize, opt.nameColor, '700')
  ctx.fillText(opt.name, opt.contentMarginLR + opt.logoSize + opt.smallFontSize, opt.contentMarginTB + Math.round(opt.logoSize / 2));
  /** UserIDを描画 */
  setFont(opt.smallFontSize, opt.userIdColor, '200')
  ctx.fillText(opt.userId, opt.contentMarginLR + opt.logoSize + opt.smallFontSize, opt.contentMarginTB + Math.round(opt.logoSize * 0.98));

  /** 時間を描画 */
  setFont(opt.smallFontSize, opt.timeColor, '200', 'right')
  const nowTime = getNowTime()
  ctx.fillText(nowTime, opt.width - opt.marginLR - opt.paddingLR, canvas.height - opt.marginTB - opt.paddingTB);

  /** アバターを描画 */
  await loadImage(opt.logo, opt.contentMarginLR, opt.contentMarginTB)
  await loadImage(opt.appLogo, canvas.width - opt.marginLR - opt.paddingLR / 2 - opt.logoSize, opt.marginTB + opt.paddingTB / 2)

  /** 出力 */
  // 1. クリップボードに出力
  if (opt.writeToClipboard) {
    await new Promise(async (reslove) => {
      canvas.toBlob(async (blob) => {
        let res = await navigator.clipboard.write([new ClipboardItem({
          [blob.type]: blob
        })]).then(() => {
          // 通知ボックス
          let notice = new tp.obsidian.Notice()
          notice.setMessage("画像がコピーされました ~")
          setTimeout(notice.hide, 2000)
        }).catch(err => {
          let notice = new tp.obsidian.Notice()
          notice.setMessage("画像のクリップボードへの書き込みに失敗しました")
          setTimeout(notice.hide, 2000)
          
          throw new Error(err)
        })

        reslove()
      })
    })
  }

  // 2. ローカルにダウンロード
  if (opt.downloadToDisk) {
    downloadImgFromCanvas(nowTime)
  }

  // 3. ドキュメントに直接書き込む
  if (opt.writeToDocument) {
    return selectedText + '\n\n' + '![](' + canvas.toDataURL('image/png') + ')'
  }

  return selectedText
}
/** Obsidian ロゴ 256*256 */
const AppLogo = ``
module.exports = get_tweet_card;
読み込み中...
文章は、創作者によって署名され、ブロックチェーンに安全に保存されています。