application_helper.rb 8.06 KB
Newer Older
gitlabhq's avatar
gitlabhq committed
1
require 'digest/md5'
Sergey Linnik's avatar
Sergey Linnik committed
2
require 'uri'
Robert Speicher's avatar
Robert Speicher committed
3

gitlabhq's avatar
gitlabhq committed
4
module ApplicationHelper
5 6
  # Check if a particular controller is the current one
  #
7 8
  # args - One or more controller names to check
  #
9 10 11
  # Examples
  #
  #   # On TreeController
12 13 14 15
  #   current_controller?(:tree)           # => true
  #   current_controller?(:commits)        # => false
  #   current_controller?(:commits, :tree) # => true
  def current_controller?(*args)
16 17 18
    args.any? do |v|
      v.to_s.downcase == controller.controller_name || v.to_s.downcase == controller.controller_path
    end
19 20
  end

Johannes Schleifenbaum's avatar
Johannes Schleifenbaum committed
21
  # Check if a particular action is the current one
Robert Speicher's avatar
Robert Speicher committed
22 23 24 25 26 27 28 29 30 31 32 33 34
  #
  # args - One or more action names to check
  #
  # Examples
  #
  #   # On Projects#new
  #   current_action?(:new)           # => true
  #   current_action?(:create)        # => false
  #   current_action?(:new, :create)  # => true
  def current_action?(*args)
    args.any? { |v| v.to_s.downcase == action_name }
  end

35
  def project_icon(project_id, options = {})
36 37
    project =
      if project_id.is_a?(Project)
38
        project_id
39 40 41 42
      else
        Project.find_with_namespace(project_id)
      end

sue445's avatar
sue445 committed
43 44
    if project.avatar_url
      image_tag project.avatar_url, options
45 46 47 48 49 50
    else # generated icon
      project_identicon(project, options)
    end
  end

  def project_identicon(project, options = {})
51 52 53 54 55 56 57 58 59 60
    allowed_colors = {
      red: 'FFEBEE',
      purple: 'F3E5F5',
      indigo: 'E8EAF6',
      blue: 'E3F2FD',
      teal: 'E0F2F1',
      orange: 'FBE9E7',
      gray: 'EEEEEE'
    }

61 62
    options[:class] ||= ''
    options[:class] << ' identicon'
63
    bg_key = project.id % 7
Gabriel Mazetto's avatar
Gabriel Mazetto committed
64
    style = "background-color: ##{allowed_colors.values[bg_key]}; color: #555"
65

66
    content_tag(:div, class: options[:class], style: style) do
67
      project.name[0, 1].upcase
68 69 70
    end
  end

Jan-Gerd Tenberge's avatar
Jan-Gerd Tenberge committed
71
  def avatar_icon(user_or_email = nil, size = nil, scale = 2)
72 73 74
    if user_or_email.is_a?(User)
      user = user_or_email
    else
75
      user = User.find_by_any_email(user_or_email.try(:downcase))
76
    end
77 78 79

    if user
      user.avatar_url(size) || default_avatar
80
    else
Jan-Gerd Tenberge's avatar
Jan-Gerd Tenberge committed
81
      gravatar_icon(user_or_email, size, scale)
82 83 84
    end
  end

85 86
  def gravatar_icon(user_email = '', size = nil, scale = 2)
    GravatarService.new.execute(user_email, size, scale) ||
87 88
      default_avatar
  end
89

90
  def default_avatar
91
    'no_avatar.png'
gitlabhq's avatar
gitlabhq committed
92 93 94
  end

  def last_commit(project)
Nihad Abbasov's avatar
Nihad Abbasov committed
95
    if project.repo_exists?
96
      time_ago_with_tooltip(project.repository.commit.committed_date)
Nihad Abbasov's avatar
Nihad Abbasov committed
97
    else
98
      'Never'
gitlabhq's avatar
gitlabhq committed
99
    end
100
  rescue
101
    'Never'
gitlabhq's avatar
gitlabhq committed
102 103
  end

104 105
  # Define whenever show last push event
  # with suggestion to create MR
randx's avatar
randx committed
106
  def show_last_push_widget?(event)
107 108 109 110 111 112 113 114 115 116 117
    # Skip if event is not about added or modified non-master branch
    return false unless event && event.last_push_to_non_root? && !event.rm_ref?

    project = event.project

    # Skip if project repo is empty or MR disabled
    return false unless project && !project.empty_repo? && project.merge_requests_enabled

    # Skip if user already created appropriate MR
    return false if project.merge_requests.where(source_branch: event.branch_name).opened.any?

118
    # Skip if user removed branch right after that
119
    return false unless project.repository.branch_exists?(event.branch_name)
120

121
    true
randx's avatar
randx committed
122
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
123

124 125 126
  def hexdigest(string)
    Digest::SHA1.hexdigest string
  end
127

128
  def simple_sanitize(str)
129 130 131
    sanitize(str, tags: %w(a span))
  end

132
  def body_data_page
133 134 135
    path = controller.controller_path.split('/')
    namespace = path.first if path.second

136
    [namespace, controller.controller_name, controller.action_name].compact.join(':')
137
  end
138 139 140 141 142 143 144 145 146 147

  # shortcut for gitlab config
  def gitlab_config
    Gitlab.config.gitlab
  end

  # shortcut for gitlab extra config
  def extra_config
    Gitlab.config.extra
  end
148

149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
  # Render a `time` element with Javascript-based relative date and tooltip
  #
  # time       - Time object
  # placement  - Tooltip placement String (default: "top")
  # html_class - Custom class for `time` element (default: "time_ago")
  # skip_js    - When true, exclude the `script` tag (default: false)
  #
  # By default also includes a `script` element with Javascript necessary to
  # initialize the `timeago` jQuery extension. If this method is called many
  # times, for example rendering hundreds of commits, it's advisable to disable
  # this behavior using the `skip_js` argument and re-initializing `timeago`
  # manually once all of the elements have been rendered.
  #
  # A `js-timeago` class is always added to the element, even when a custom
  # `html_class` argument is provided.
  #
  # Returns an HTML-safe String
  def time_ago_with_tooltip(time, placement: 'top', html_class: 'time_ago', skip_js: false)
    element = content_tag :time, time.to_s,
168
      class: "#{html_class} js-timeago #{"js-timeago-pending" unless skip_js}",
169
      datetime: time.to_time.getutc.iso8601,
170
      title: time.to_time.in_time_zone.to_s(:medium),
171
      data: { toggle: 'tooltip', placement: placement, container: 'body' }
172

173 174 175 176 177
    unless skip_js
      element << javascript_tag(
        "$('.js-timeago-pending').removeClass('js-timeago-pending').timeago()"
      )
    end
178 179

    element
180
  end
181

182 183
  def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', include_author: false)
    return if object.updated_at == object.created_at
184 185

    content_tag :small, class: "edited-text" do
186 187
      output = content_tag(:span, "Edited ")
      output << time_ago_with_tooltip(object.updated_at, placement: placement, html_class: html_class)
188

Phil Hughes's avatar
Phil Hughes committed
189 190 191
      if include_author && object.updated_by && object.updated_by != object.author
        output << content_tag(:span, " by ")
        output << link_to_member(object.project, object.updated_by, avatar: false, author_class: nil)
192 193 194 195 196 197
      end

      output
    end
  end

198
  def render_markup(file_name, file_content)
199
    if gitlab_markdown?(file_name)
connorshea's avatar
connorshea committed
200
      Hamlit::RailsHelpers.preserve(markdown(file_content))
201
    elsif asciidoc?(file_name)
202
      asciidoc(file_content)
203 204 205 206
    elsif plain?(file_name)
      content_tag :pre, class: 'plain-readme' do
        file_content
      end
207
    else
208
      other_markup(file_name, file_content)
209
    end
210 211
  rescue RuntimeError
    simple_format(file_content)
212
  end
213

214 215 216 217
  def plain?(filename)
    Gitlab::MarkupHelper.plain?(filename)
  end

218
  def markup?(filename)
219
    Gitlab::MarkupHelper.markup?(filename)
220 221 222
  end

  def gitlab_markdown?(filename)
223
    Gitlab::MarkupHelper.gitlab_markdown?(filename)
224 225
  end

226
  def asciidoc?(filename)
227
    Gitlab::MarkupHelper.asciidoc?(filename)
228 229
  end

230 231 232 233 234 235 236
  def promo_host
    'about.gitlab.com'
  end

  def promo_url
    'https://' + promo_host
  end
237

238 239
  def page_filter_path(options = {})
    without = options.delete(:without)
Arinde Eniola's avatar
Arinde Eniola committed
240
    add_label = options.delete(:label)
241

242 243 244
    exist_opts = {
      state: params[:state],
      scope: params[:scope],
245
      milestone_title: params[:milestone_title],
246 247 248
      assignee_id: params[:assignee_id],
      author_id: params[:author_id],
      sort: params[:sort],
249 250
      issue_search: params[:issue_search],
      label_name: params[:label_name]
251 252 253 254
    }

    options = exist_opts.merge(options)

255 256 257 258 259 260
    if without.present?
      without.each do |key|
        options.delete(key)
      end
    end

Phil Hughes's avatar
Phil Hughes committed
261
    params = options.compact
262

263
    params.delete(:label_name) unless add_label
264

265
    "#{request.path}?#{params.to_param}"
266
  end
267 268 269 270

  def outdated_browser?
    browser.ie? && browser.version.to_i < 10
  end
271 272 273 274 275 276 277 278

  def path_to_key(key, admin = false)
    if admin
      admin_user_key_path(@user, key)
    else
      profile_key_path(key)
    end
  end
279

280
  def state_filters_text_for(entity, project)
281
    titles = {
282
      opened: "Open"
283
    }
284

285
    entity_title = titles[entity] || entity.to_s.humanize
286 287

    count =
288
      if project.nil?
289
        nil
290
      elsif current_controller?(:issues)
291
        project.issues.visible_to_user(current_user).send(entity).count
292
      elsif current_controller?(:merge_requests)
293
        project.merge_requests.send(entity).count
294
      end
295

296
    html = content_tag :span, entity_title
297 298

    if count.present?
299
      html += " "
300 301 302 303
      html += content_tag :span, number_with_delimiter(count), class: 'badge'
    end

    html.html_safe
304
  end
305 306 307 308

  def truncate_first_line(message, length = 50)
    truncate(message.each_line.first.chomp, length: length) if message
  end
gitlabhq's avatar
gitlabhq committed
309
end