module Nanoc::Helpers
# Contains functions for linking to items and item representations.
module LinkTo
require 'nanoc/helpers/html_escape'
include Nanoc::Helpers::HTMLEscape
# Creates a HTML link to the given path or item representation, and with
# the given text. All attributes of the `a` element, including the `href`
# attribute, will be HTML-escaped; the contents of the `a` element, which
# can contain markup, will not be HTML-escaped. The HTML-escaping is done
# using {Nanoc::Helpers::HTMLEscape#html_escape}.
#
# @param [String] text The visible link text
#
# @param [String, Nanoc::Int::Item, Nanoc::Int::ItemRep] target The path/URL,
# item or item representation that should be linked to
#
# @param [Hash] attributes A hash containing HTML attributes (e.g.
# `rel`, `title`, …) that will be added to the link.
#
# @return [String] The link text
#
# @example Linking to a path
#
# link_to('Blog', '/blog/')
# # => 'Blog'
#
# @example Linking to an item
#
# about = @items.find { |i| i.identifier == '/about/' }
# link_to('About Me', about)
# # => 'About Me'
#
# @example Linking to an item representation
#
# about = @items.find { |i| i.identifier == '/about/' }
# link_to('My vCard', about.rep(:vcard))
# # => 'My vCard'
#
# @example Linking with custom attributes
#
# link_to('Blog', '/blog/', :title => 'My super cool blog')
# # => 'Blog'
def link_to(text, target, attributes = {})
# Find path
if target.is_a?(String)
path = target
else
path = target.path
raise "Cannot create a link to #{target.inspect} because this target is not outputted (its routing rule returns nil)" if path.nil?
end
# Join attributes
attributes = attributes.reduce('') do |memo, (key, value)|
memo + key.to_s + '="' + h(value) + '" '
end
# Create link
"#{text}"
end
# Creates a HTML link using {#link_to}, except when the linked item is
# the current one. In this case, a span element with class “active” and
# with the given text will be returned. The HTML-escaping rules for
# {#link_to} apply here as well.
#
# @param [String] text The visible link text
#
# @param [String, Nanoc::Int::Item, Nanoc::Int::ItemRep] target The path/URL,
# item or item representation that should be linked to
#
# @param [Hash] attributes A hash containing HTML attributes (e.g.
# `rel`, `title`, …) that will be added to the link.
#
# @return [String] The link text
#
# @example Linking to a different page
#
# link_to_unless_current('Blog', '/blog/')
# # => 'Blog'
#
# @example Linking to the same page
#
# link_to_unless_current('This Item', @item)
# # => 'This Item'
def link_to_unless_current(text, target, attributes = {})
# Find path
path = target.is_a?(String) ? target : target.path
if @item_rep && @item_rep.path == path
# Create message
"#{text}"
else
link_to(text, target, attributes)
end
end
# Returns the relative path from the current item to the given path or
# item representation. The returned path will not be HTML-escaped.
#
# @param [String, Nanoc::Int::Item, Nanoc::Int::ItemRep] target The path/URL,
# item or item representation to which the relative path should be
# generated
#
# @return [String] The relative path to the target
#
# @example
#
# # if the current item's path is /foo/bar/
# relative_path_to('/foo/qux/')
# # => '../qux/'
def relative_path_to(target)
require 'pathname'
# Find path
if target.is_a?(String)
path = target
else
path = target.path
if path.nil?
raise "Cannot get the relative path to #{target.inspect} because this target is not outputted (its routing rule returns nil)"
end
end
# Handle Windows network (UNC) paths
if path.start_with?('//') || path.start_with?('\\\\')
return path
end
# Get source and destination paths
dst_path = Pathname.new(path)
if @item_rep.path.nil?
raise "Cannot get the relative path to #{path} because the current item representation, #{@item_rep.inspect}, is not outputted (its routing rule returns nil)"
end
src_path = Pathname.new(@item_rep.path)
# Calculate the relative path (method depends on whether destination is
# a directory or not).
if src_path.to_s[-1, 1] != '/'
relative_path = dst_path.relative_path_from(src_path.dirname).to_s
else
relative_path = dst_path.relative_path_from(src_path).to_s
end
# Add trailing slash if necessary
if dst_path.to_s[-1, 1] == '/'
relative_path << '/'
end
# Done
relative_path
end
end
end