gitlab-org--gitlab-foss/app/services/task_list_toggle_service.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

75 lines
2.5 KiB
Ruby
Raw Normal View History

2019-01-24 22:13:35 +00:00
# frozen_string_literal: true
# Finds the correct checkbox in the passed in markdown/html and toggles it's state,
2019-01-25 18:50:21 +00:00
# returning the updated markdown/html.
2019-01-24 22:13:35 +00:00
# We don't care if the text has changed above or below the specific checkbox, as long
2019-01-25 18:50:21 +00:00
# the checkbox still exists at exactly the same line number and the text is equal.
2019-01-24 22:13:35 +00:00
# If successful, new values are available in `updated_markdown` and `updated_markdown_html`
class TaskListToggleService
attr_reader :updated_markdown, :updated_markdown_html
def initialize(markdown, markdown_html, line_source:, line_number:, toggle_as_checked:)
@markdown = markdown
@markdown_html = markdown_html
@line_source = line_source
@line_number = line_number
@toggle_as_checked = toggle_as_checked
2019-01-24 22:13:35 +00:00
@updated_markdown, @updated_markdown_html = nil
end
def execute
return false unless markdown && markdown_html
2019-01-31 15:33:38 +00:00
toggle_markdown && toggle_markdown_html
2019-01-24 22:13:35 +00:00
end
private
attr_reader :markdown, :markdown_html, :toggle_as_checked
attr_reader :line_source, :line_number
2019-01-24 22:13:35 +00:00
def toggle_markdown
2019-01-29 19:11:04 +00:00
source_lines = markdown.split("\n")
source_line_index = line_number - 1
markdown_task = source_lines[source_line_index]
2019-01-24 22:13:35 +00:00
# The source in the DB could be using either \n or \r\n line endings
2019-02-07 20:51:11 +00:00
return unless markdown_task.chomp == line_source
2019-01-24 22:13:35 +00:00
return unless source_checkbox = Taskable::ITEM_PATTERN.match(markdown_task)
currently_checked = TaskList::Item.new(source_checkbox[2]).complete?
2019-01-29 19:11:04 +00:00
# Check `toggle_as_checked` to make sure we don't accidentally replace
# any `[ ]` or `[x]` in the middle of the text
if currently_checked
markdown_task.sub!(Taskable::COMPLETE_PATTERN, '[ ]') unless toggle_as_checked
2019-01-24 22:13:35 +00:00
else
2019-01-29 19:11:04 +00:00
markdown_task.sub!(Taskable::INCOMPLETE_PATTERN, '[x]') if toggle_as_checked
2019-01-24 22:13:35 +00:00
end
2019-01-29 19:11:04 +00:00
source_lines[source_line_index] = markdown_task
2019-01-24 22:13:35 +00:00
@updated_markdown = source_lines.join("\n")
end
2019-01-29 19:11:04 +00:00
def toggle_markdown_html
2019-01-24 22:13:35 +00:00
html = Nokogiri::HTML.fragment(markdown_html)
2019-01-25 18:50:21 +00:00
html_checkbox = get_html_checkbox(html)
2019-01-24 22:13:35 +00:00
return unless html_checkbox
2019-01-29 19:11:04 +00:00
if toggle_as_checked
2019-01-24 22:13:35 +00:00
html_checkbox[:checked] = 'checked'
2019-01-29 19:11:04 +00:00
else
html_checkbox.remove_attribute('checked')
2019-01-24 22:13:35 +00:00
end
@updated_markdown_html = html.to_html
end
2019-01-25 18:50:21 +00:00
# When using CommonMark, we should be able to use the embedded `sourcepos` attribute to
# target the exact line in the DOM.
2019-01-25 18:50:21 +00:00
def get_html_checkbox(html)
html.css(".task-list-item[data-sourcepos^='#{line_number}:'] input.task-list-item-checkbox").first
2019-01-25 18:50:21 +00:00
end
2019-01-24 22:13:35 +00:00
end