Webサービスやアプリを開発して運営するのが趣味です。



ブログにシンタックスハイライト機能つけた

2013年02月20日 23:40

ブログにコードを載せることが多くなりそうなので、コードスニペットをいい感じにHTMLで成形してくれるやつを組み込んでみた。1年前の自分はsyntaxhighlight.inがよさそうなんて言ってるけど、いちいちWebサイト開いてコピペする作業するのはだいぶない思う。

このブログはRailsで組んでてMarkdownで書けるようになっているので、ヘルパーでいい感じにやってくれるようにします。

application_helper.rb


module ApplicationHelper
def markdown(text)
html = Redcarpet::Markdown.new(
Redcarpet::Render::HTML.new(hard_wrap: true),
tables: true,
autolink: true,
superscript: true,
strikethrough: true,
no_intra_emphasis: true,
fenced_code_blocks: true,
space_after_headers: true
).render(text)
syntax_highlighter(html).html_safe
end

def syntax_highlighter(html)
doc = Nokogiri::HTML(html)
doc.search("pre").each do |pre|
pre.replace(Pygments.highlight(
pre.text.rstrip,
lexer: pre.children.attribute("class").value
))
end
doc.to_s
end
end

Markdownの解釈にRedcarpet、構文の解釈にpygments.rbを使用する故、その旨をGemfileに記載しておきます。

Gemfile


gem "redcarpet"
gem "pygments.rb"
gem "nokogiri"

これだけで markdown(@post.body) みたいにすればガリガリHTML組んでくれるので、あとはCSSを適用するだけです。Githubの色がとても好きなのでrichleland/pygments-cssからGithub風のCSSを頂戴して少し手を加えたやつを使うことにしました。

pygments.css.scss


.highlight {
font-family: Consolas,"Liberation Mono",Courier,monospace;
font-size: 12px;
line-height: 1.4;
padding: 12px;
padding-right: 0;
margin-bottom: 16px;
border: solid 2px #eee;
border-radius: 2px;
background-color: #fafafa;
overflow: scroll;

.hll { background-color: #ffffcc }
.c { color: #999988; font-style: italic } /* Comment */
.err { color: #a61717; background-color: #e3d2d2 } /* Error */
.k { color: #000000; font-weight: bold } /* Keyword */
.o { color: #000000; font-weight: bold } /* Operator */
.cm { color: #999988; font-style: italic } /* Comment.Multiline */
.cp { color: #999999; font-weight: bold; font-style: italic } /* Comment.Preproc */
.c1 { color: #999988; font-style: italic } /* Comment.Single */
.cs { color: #999999; font-weight: bold; font-style: italic } /* Comment.Special */
.gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.ge { color: #000000; font-style: italic } /* Generic.Emph */
.gr { color: #aa0000 } /* Generic.Error */
.gh { color: #999999 } /* Generic.Heading */
.gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.go { color: #888888 } /* Generic.Output */
.gp { color: #555555 } /* Generic.Prompt */
.gs { font-weight: bold } /* Generic.Strong */
.gu { color: #aaaaaa } /* Generic.Subheading */
.gt { color: #aa0000 } /* Generic.Traceback */
.kc { color: #000000; font-weight: bold } /* Keyword.Constant */
.kd { color: #000000; font-weight: bold } /* Keyword.Declaration */
.kn { color: #000000; font-weight: bold } /* Keyword.Namespace */
.kp { color: #000000; font-weight: bold } /* Keyword.Pseudo */
.kr { color: #000000; font-weight: bold } /* Keyword.Reserved */
.kt { color: #445588; font-weight: bold } /* Keyword.Type */
.m { color: #009999 } /* Literal.Number */
.s { color: #d01040 } /* Literal.String */
.na { color: #008080 } /* Name.Attribute */
.nb { color: #0086B3 } /* Name.Builtin */
.nc { color: #445588; font-weight: bold } /* Name.Class */
.no { color: #008080 } /* Name.Constant */
.nd { color: #3c5d5d; font-weight: bold } /* Name.Decorator */
.ni { color: #800080 } /* Name.Entity */
.ne { color: #990000; font-weight: bold } /* Name.Exception */
.nf { color: #990000; font-weight: bold } /* Name.Function */
.nl { color: #990000; font-weight: bold } /* Name.Label */
.nn { color: #555555 } /* Name.Namespace */
.nt { color: #000080 } /* Name.Tag */
.nv { color: #008080 } /* Name.Variable */
.ow { color: #000000; font-weight: bold } /* Operator.Word */
.w { color: #bbbbbb } /* Text.Whitespace */
.mf { color: #009999 } /* Literal.Number.Float */
.mh { color: #009999 } /* Literal.Number.Hex */
.mi { color: #009999 } /* Literal.Number.Integer */
.mo { color: #009999 } /* Literal.Number.Oct */
.sb { color: #d01040 } /* Literal.String.Backtick */
.sc { color: #d01040 } /* Literal.String.Char */
.sd { color: #d01040 } /* Literal.String.Doc */
.s2 { color: #d01040 } /* Literal.String.Double */
.se { color: #d01040 } /* Literal.String.Escape */
.sh { color: #d01040 } /* Literal.String.Heredoc */
.si { color: #d01040 } /* Literal.String.Interpol */
.sx { color: #d01040 } /* Literal.String.Other */
.sr { color: #009926 } /* Literal.String.Regex */
.s1 { color: #d01040 } /* Literal.String.Single */
.ss { color: #990073 } /* Literal.String.Symbol */
.bp { color: #999999 } /* Name.Builtin.Pseudo */
.vc { color: #008080 } /* Name.Variable.Class */
.vg { color: #008080 } /* Name.Variable.Global */
.vi { color: #008080 } /* Name.Variable.Instance */
.il { color: #009999 } /* Literal.Number.Integer.Long */
}

満足度高い