.. _homework_html_renderer: ================================== HTML Renderer Homework Assignment ================================== HTML Render ============ Goal: ------ The goal is to create a set of classes to render html pages -- in a "pretty printed" way. i.e nicely indented and human readable. We'll try to get to all the features required to render: :download:`sample_html.html <./sample_html.html>` The exercise is broken down into a number of steps -- each requiring a few more OO concepts in Python. General Instructions: --------------------- For each step, add the required functionality. There is example code to run your code for each step in: ``code\session06\run_html_render.py`` name your file: ``html_render.py`` -- so it can be imported by ``run_html_render.py`` You should be able to run that code at each step, uncommenting each new step in ``run_html_render.py`` as you go. It builds up a html tree, and then calls the ``render()`` method of your element to render the page. It uses a ``cStringIO`` object (like a file, but in memory) to render to memory, then dumps it to the console, and writes a file. Take a look at the code at the end to make sure you understand it. The html generated at each step is in the files: ``test_html_ouput?.html`` At each step, your results should look similar that those (maybe not identical...) Step 1: ------- Create an ``Element`` class for rendering an html element (xml element). It should have class attributes for the tag name ("html" first) and the indentation (spaces to indent for pretty printing) The constructor signature should look like .. code-block:: python Element(content=None) where ``content`` is a string It should have an ``append`` method that can add another string to the content. It should have a ``render(file_out, ind = "")`` method that renders the tag and the strings in the content. ``file_out`` could be any file-like object ( i.e. have a ``write()`` method ). .. nextslide:: ``ind`` is a string with the indentation level in it: the amount that the tag should be indented for pretty printing. - This is a little tricky: ``ind`` will be the amount that this element should be indented already. It will be from zero (an empty string) to a lot of spaces, depending on how deep it is in the tree. The amount of indentation should be set by the class attribute: ``indent`` You should now be able to render an html tag with text in it as contents. See: step 1. in ``run_html_render.py`` Step 2: -------- Create a couple subclasses of ``Element``, for a ``
`` tag and ```` tag. All you should have to do is override the ``tag`` class attribute (you may need to add a ``tag`` class attribute to the Element class first...). Now you can render a few different types of element. Extend the ``Element.render()`` method so that it can render other elements inside the tag in addition to strings. Simple recursion should do it. i.e. it can call the ``render()`` method of the elements it contains. You'll need to be smart about setting the ``ind`` optional parameter -- so that the nested elements get indented correctly. Figure out a way to deal with the fact that the contained elements could be either simple strings or ``Element`` s with render methods (there are a few ways to handle that...). You should now be able to render a basic web page with an html tag around the whole thing, a ``
`` tag inside, and multiple ```` tags inside that, with text inside that. And all indended nicely. See ``test_html_output2.html`` Step 3: -------- Create a ``
`` element -- simple subclass. Create a ``OneLineTag`` subclass of ``Element``: * It should override the render method, to render everything on one line -- for the simple tags, like::`` elements and some text. See ``test_html_output3.html`` Step 4: -------- Extend the ``Element`` class to accept a set of attributes as keywords to the constructor, ie. (``run_html_render.py``) .. code-block:: python Element("some text content", id="TheList", style="line-height:200%") ( remember ``**kwargs``? ) The render method will need to be extended to render the attributes properly. You can now render some ``
`` tags (and others) with attributes See ``test_html_output4.html`` Step 5: -------- Create a ``SelfClosingTag`` subclass of Element, to render tags like::