From 2cfc8c4c5d38b03946beaff49761cf57bcdc7d58 Mon Sep 17 00:00:00 2001 From: Marc Wouts Date: Wed, 22 Jul 2020 00:38:54 +0200 Subject: [PATCH] Cell marker has the same indentation as the first line in the cell #562 --- CHANGELOG.md | 1 + jupytext/cell_reader.py | 4 ++-- jupytext/cell_to_text.py | 12 ++++++++++-- tests/test_read_simple_percent.py | 29 +++++++++++++++++++++++++++++ 4 files changed, 42 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index eb0fa98a0..732c405f1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,7 @@ **Changed** - Install Jupytext from source on MyBinder to avoid cache issues (#567) - Skip the tests that execute a notebook on Windows to avoid timeout issues (#489) +- The `# %%` cell marker has the same indentation as the first line in the cell (#562) **Fixed** - Configured coverage targets in `codecov.yml` diff --git a/jupytext/cell_reader.py b/jupytext/cell_reader.py index b3d3de74f..627d58b2c 100644 --- a/jupytext/cell_reader.py +++ b/jupytext/cell_reader.py @@ -747,9 +747,9 @@ def __init__(self, fmt, default_language=None): script = _SCRIPT_EXTENSIONS[self.ext] self.default_language = default_language or script["language"] self.comment = script["comment"] - self.start_code_re = re.compile(r"^{}\s*%%(%*)\s(.*)$".format(self.comment)) + self.start_code_re = re.compile(r"^\s*{}\s*%%(%*)\s(.*)$".format(self.comment)) self.alternative_start_code_re = re.compile( - r"^{}\s*(%%||In\[[0-9 ]*\]:?)\s*$".format(self.comment) + r"^\s*{}\s*(%%||In\[[0-9 ]*\]:?)\s*$".format(self.comment) ) self.explicit_soc = True diff --git a/jupytext/cell_to_text.py b/jupytext/cell_to_text.py index 369036c4f..b9a4f9acf 100644 --- a/jupytext/cell_to_text.py +++ b/jupytext/cell_to_text.py @@ -457,10 +457,18 @@ def cell_to_text(self): options = metadata_to_double_percent_options( self.metadata, self.cell_metadata_json ) + indent = "" + if self.is_code() and active and self.source: + first_line = self.source[0] + if first_line.strip(): + left_space = re.compile(r"^(\s*)").match(first_line) + if left_space: + indent = left_space.groups()[0] + if options.startswith("%") or not options: - lines = [self.comment + " %%" + options] + lines = [indent + self.comment + " %%" + options] else: - lines = [self.comment + " %% " + options] + lines = [indent + self.comment + " %% " + options] if self.is_code() and active: source = copy(self.source) diff --git a/tests/test_read_simple_percent.py b/tests/test_read_simple_percent.py index 60839e8c7..bfc79ff40 100644 --- a/tests/test_read_simple_percent.py +++ b/tests/test_read_simple_percent.py @@ -502,3 +502,32 @@ def test_docstring_with_quadruple_quote( py = jupytext.writes(nb, "py:percent") nb2 = jupytext.reads(py, "py") compare_notebooks(nb2, nb) + + +def test_cell_marker_has_same_indentation_as_code( + text="""# %% +if __name__ == '__main__': + print(1) + + # %% + # INDENTED COMMENT + print(2) +""", + nb_expected=new_notebook( + cells=[ + new_code_cell( + """if __name__ == '__main__': + print(1)""" + ), + new_code_cell( + """ # INDENTED COMMENT + print(2)""" + ), + ] + ), +): + """The cell marker should have the same indentation as the first code line. See issue #562""" + nb_actual = jupytext.reads(text, fmt="py:percent") + compare_notebooks(nb_actual, nb_expected) + text_actual = jupytext.writes(nb_actual, fmt="py:percent") + compare(text_actual, text)