Skip to content

Package drawing

This module provides visualisation of loci annotation.

ColorLegendVis

Bases: Track

ColorLegend track object that handles visualisation of legend to feature's category colours.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class ColorLegendVis(Track):
    """ColorLegend track object that handles visualisation of legend to feature's category colours.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        prms (lovis4u.Manager.Parameters): Parameters' class object.

    """

    def __init__(self, layout: dict, track_data: dict, parameters):
        """Create a ColorLegend object.

        Arguments:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            track_data (dict): a dictionary with prepared track specific data.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.

        """
        super().__init__(layout, track_data, parameters)
        self.track_height = None

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Draw a ColorLegend track.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        try:
            y_upper = self.layout["current_y_coordinate"]
            canvas.setLineWidth(self.prms.args["scale_line_width"])
            canvas.setFont(self.prms.args["colour_legend_font_face"],
                           self.track_data["colour_legend_label_size"])
            for label_dict in self.track_data["labels"]:
                yl = y_upper + label_dict["relative_y"]
                yt = y_upper + label_dict["relative_y_text"]
                canvas.setFillColorRGB(*matplotlib.colors.hex2color(label_dict["colour"]),
                                       self.prms.args["category_annotation_alpha"])
                canvas.rect(label_dict["label_x"], yl, label_dict["label_width"],
                            self.track_data["line_width"], fill=1, stroke=0)
                canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("colour_legend_label_colour", self.prms))
                canvas.drawString(label_dict["label_x"], yt, label_dict["label"])
            return None
        except Exception as error:
            raise lovis4u.Manager.lovis4uError("Unable to draw a colour legend track.") from error

__init__(layout, track_data, parameters)

Create a ColorLegend object.

Parameters:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout: dict, track_data: dict, parameters):
    """Create a ColorLegend object.

    Arguments:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.

    """
    super().__init__(layout, track_data, parameters)
    self.track_height = None

draw(canvas)

Draw a ColorLegend track.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Draw a ColorLegend track.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    try:
        y_upper = self.layout["current_y_coordinate"]
        canvas.setLineWidth(self.prms.args["scale_line_width"])
        canvas.setFont(self.prms.args["colour_legend_font_face"],
                       self.track_data["colour_legend_label_size"])
        for label_dict in self.track_data["labels"]:
            yl = y_upper + label_dict["relative_y"]
            yt = y_upper + label_dict["relative_y_text"]
            canvas.setFillColorRGB(*matplotlib.colors.hex2color(label_dict["colour"]),
                                   self.prms.args["category_annotation_alpha"])
            canvas.rect(label_dict["label_x"], yl, label_dict["label_width"],
                        self.track_data["line_width"], fill=1, stroke=0)
            canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("colour_legend_label_colour", self.prms))
            canvas.drawString(label_dict["label_x"], yt, label_dict["label"])
        return None
    except Exception as error:
        raise lovis4u.Manager.lovis4uError("Unable to draw a colour legend track.") from error

CrossTrack

Parent class for Cross-Tracks visualisation.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • tracks (dict) –

    List with track objects participated in CrossTrack.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class CrossTrack:
    """Parent class for Cross-Tracks visualisation.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        tracks (dict): List with track objects participated in CrossTrack.
        prms (lovis4u.Manager.Parameters): Parameters' class object.

    """

    def __init__(self, layout, tracks, parameters):
        """Parent's constructor for creating a CrossTrack object.

        Attributes:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            tracks (dict): List with track objects participated in CrossTrack.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.

        """
        self.layout = layout
        self.tracks = tracks
        self.prms = parameters

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas):
        """Empy parent's method for cross track drawing.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        pass

__init__(layout, tracks, parameters)

Parent's constructor for creating a CrossTrack object.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • tracks (dict) –

    List with track objects participated in CrossTrack.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout, tracks, parameters):
    """Parent's constructor for creating a CrossTrack object.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        tracks (dict): List with track objects participated in CrossTrack.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.

    """
    self.layout = layout
    self.tracks = tracks
    self.prms = parameters

draw(canvas)

Empy parent's method for cross track drawing.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas):
    """Empy parent's method for cross track drawing.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    pass

HomologyTrack

Bases: CrossTrack

Track that handle visualisation of homology lines between homologous features on neighbours' loci.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • tracks (dict) –

    List with track objects participated in CrossTrack.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class HomologyTrack(CrossTrack):
    """Track that handle visualisation of homology lines between homologous features on neighbours' loci.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        tracks (dict): List with track objects participated in CrossTrack.
        prms (lovis4u.Manager.Parameters): Parameters' class object.

    """

    def __init__(self, layout, tracks, parameters):
        """Create a HomologyTrack.

        Attributes:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            tracks (dict): List with track objects participated in CrossTrack.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.

        """
        super().__init__(layout, tracks, parameters)

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Draw HomologyTrack on canvas.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        try:
            for track in self.tracks:
                track.track_data["y_top"] = self.layout["figure_height"] - track.layout["inverse_y_coordinate"]
                track.track_data["feature_upper"] = track.track_data["y_top"] - \
                                                    (track.track_data["n_label_rows"] * track.track_data[
                                                        "f_label_height"] *
                                                     (1 + track.prms.args["feature_label_gap"]))
            num_of_loci_tracks = len(self.tracks)
            for ti in range(num_of_loci_tracks - 1):
                current_track = self.tracks[ti]
                current_track_features = current_track.track_data["features"]
                next_track = self.tracks[ti + 1]
                next_track_features = next_track.track_data["features"]
                for ctf in current_track_features:
                    ctf_group = ctf["group"]
                    next_track_same_group_features = [i for i in next_track_features if i["group"] == ctf_group]
                    for ntf in next_track_same_group_features:
                        cty_u = current_track.track_data["feature_upper"]
                        cty_c = current_track.track_data["feature_upper"] - 0.5 * self.prms.args["feature_height"] * mm
                        cty_b = current_track.track_data["feature_upper"] - self.prms.args["feature_height"] * mm
                        nty_u = next_track.track_data["feature_upper"]
                        nty_c = next_track.track_data["feature_upper"] - 0.5 * self.prms.args["feature_height"] * mm
                        nty_b = next_track.track_data["feature_upper"] - self.prms.args["feature_height"] * mm
                        ct_arrow_len = min(
                            self.prms.args["feature_height"] * mm * self.prms.args["feature_arrow_length"],
                            (ctf["coordinates"]["end"] - ctf["coordinates"]["start"]))
                        nt_arrow_len = min(
                            self.prms.args["feature_height"] * mm * self.prms.args["feature_arrow_length"],
                            (ntf["coordinates"]["end"] - ntf["coordinates"]["start"]))
                        canvas.setLineCap(0)
                        canvas.setLineJoin(1)
                        canvas.setLineWidth(self.prms.args["homology_line_width"])
                        canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("homology_stroke_colour", self.prms))
                        canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("homology_fill_colour", self.prms))
                        p = canvas.beginPath()

                        if ctf["coordinates"]["orient"] == 1:
                            cts, cte = ctf["coordinates"]["start"], ctf["coordinates"]["end"]
                            p.moveTo(ctf["coordinates"]["start"], cty_b)
                            if not ctf["coordinates"]["rout"]:
                                p.lineTo(ctf["coordinates"]["end"] - ct_arrow_len, cty_b)
                                p.lineTo(ctf["coordinates"]["end"], cty_c)
                            else:
                                p.lineTo(ctf["coordinates"]["end"], cty_b)
                        elif ctf["coordinates"]["orient"] == -1:
                            cts, cte = ctf["coordinates"]["end"], ctf["coordinates"]["start"]
                            p.moveTo(ctf["coordinates"]["end"], cty_b)
                            if not ctf["coordinates"]["lout"]:
                                p.lineTo(ctf["coordinates"]["start"] + ct_arrow_len, cty_b)
                                p.lineTo(ctf["coordinates"]["start"], cty_c)
                            else:
                                p.lineTo(ctf["coordinates"]["start"], cty_b)
                        if ntf["coordinates"]["orient"] == 1:
                            nts, nte = ntf["coordinates"]["end"], ntf["coordinates"]["start"]
                            if not ntf["coordinates"]["rout"]:
                                p.lineTo(ntf["coordinates"]["end"], nty_c)
                                p.lineTo(ntf["coordinates"]["end"] - nt_arrow_len, nty_u)
                            else:
                                p.lineTo(ntf["coordinates"]["end"], nty_u)
                            p.lineTo(ntf["coordinates"]["start"], nty_u)
                            if nte >= cts:
                                p.lineTo(ntf["coordinates"]["start"], nty_c)
                        elif ntf["coordinates"]["orient"] == -1:
                            nts, nte = ntf["coordinates"]["start"], ntf["coordinates"]["end"]
                            if not ntf["coordinates"]["lout"]:
                                p.lineTo(ntf["coordinates"]["start"], nty_c)
                                p.lineTo(ntf["coordinates"]["start"] + nt_arrow_len, nty_u)
                            else:
                                p.lineTo(ntf["coordinates"]["start"], nty_u)
                            p.lineTo(ntf["coordinates"]["end"], nty_u)
                            if nte <= cts:
                                p.lineTo(ntf["coordinates"]["end"], nty_c)
                        if (nte <= cts and ctf["coordinates"]["orient"] == 1) or (
                                nte >= cts and ctf["coordinates"]["orient"] == -1):
                            p.lineTo(cts, cty_c)
                        p.lineTo(cts, cty_b)
                        canvas.drawPath(p, stroke=1, fill=1)
        except Exception as error:
            raise lovis4u.Manager.lovis4uError("Unable to draw homology line track.") from error

__init__(layout, tracks, parameters)

Create a HomologyTrack.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • tracks (dict) –

    List with track objects participated in CrossTrack.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout, tracks, parameters):
    """Create a HomologyTrack.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        tracks (dict): List with track objects participated in CrossTrack.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.

    """
    super().__init__(layout, tracks, parameters)

draw(canvas)

Draw HomologyTrack on canvas.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Draw HomologyTrack on canvas.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    try:
        for track in self.tracks:
            track.track_data["y_top"] = self.layout["figure_height"] - track.layout["inverse_y_coordinate"]
            track.track_data["feature_upper"] = track.track_data["y_top"] - \
                                                (track.track_data["n_label_rows"] * track.track_data[
                                                    "f_label_height"] *
                                                 (1 + track.prms.args["feature_label_gap"]))
        num_of_loci_tracks = len(self.tracks)
        for ti in range(num_of_loci_tracks - 1):
            current_track = self.tracks[ti]
            current_track_features = current_track.track_data["features"]
            next_track = self.tracks[ti + 1]
            next_track_features = next_track.track_data["features"]
            for ctf in current_track_features:
                ctf_group = ctf["group"]
                next_track_same_group_features = [i for i in next_track_features if i["group"] == ctf_group]
                for ntf in next_track_same_group_features:
                    cty_u = current_track.track_data["feature_upper"]
                    cty_c = current_track.track_data["feature_upper"] - 0.5 * self.prms.args["feature_height"] * mm
                    cty_b = current_track.track_data["feature_upper"] - self.prms.args["feature_height"] * mm
                    nty_u = next_track.track_data["feature_upper"]
                    nty_c = next_track.track_data["feature_upper"] - 0.5 * self.prms.args["feature_height"] * mm
                    nty_b = next_track.track_data["feature_upper"] - self.prms.args["feature_height"] * mm
                    ct_arrow_len = min(
                        self.prms.args["feature_height"] * mm * self.prms.args["feature_arrow_length"],
                        (ctf["coordinates"]["end"] - ctf["coordinates"]["start"]))
                    nt_arrow_len = min(
                        self.prms.args["feature_height"] * mm * self.prms.args["feature_arrow_length"],
                        (ntf["coordinates"]["end"] - ntf["coordinates"]["start"]))
                    canvas.setLineCap(0)
                    canvas.setLineJoin(1)
                    canvas.setLineWidth(self.prms.args["homology_line_width"])
                    canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("homology_stroke_colour", self.prms))
                    canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("homology_fill_colour", self.prms))
                    p = canvas.beginPath()

                    if ctf["coordinates"]["orient"] == 1:
                        cts, cte = ctf["coordinates"]["start"], ctf["coordinates"]["end"]
                        p.moveTo(ctf["coordinates"]["start"], cty_b)
                        if not ctf["coordinates"]["rout"]:
                            p.lineTo(ctf["coordinates"]["end"] - ct_arrow_len, cty_b)
                            p.lineTo(ctf["coordinates"]["end"], cty_c)
                        else:
                            p.lineTo(ctf["coordinates"]["end"], cty_b)
                    elif ctf["coordinates"]["orient"] == -1:
                        cts, cte = ctf["coordinates"]["end"], ctf["coordinates"]["start"]
                        p.moveTo(ctf["coordinates"]["end"], cty_b)
                        if not ctf["coordinates"]["lout"]:
                            p.lineTo(ctf["coordinates"]["start"] + ct_arrow_len, cty_b)
                            p.lineTo(ctf["coordinates"]["start"], cty_c)
                        else:
                            p.lineTo(ctf["coordinates"]["start"], cty_b)
                    if ntf["coordinates"]["orient"] == 1:
                        nts, nte = ntf["coordinates"]["end"], ntf["coordinates"]["start"]
                        if not ntf["coordinates"]["rout"]:
                            p.lineTo(ntf["coordinates"]["end"], nty_c)
                            p.lineTo(ntf["coordinates"]["end"] - nt_arrow_len, nty_u)
                        else:
                            p.lineTo(ntf["coordinates"]["end"], nty_u)
                        p.lineTo(ntf["coordinates"]["start"], nty_u)
                        if nte >= cts:
                            p.lineTo(ntf["coordinates"]["start"], nty_c)
                    elif ntf["coordinates"]["orient"] == -1:
                        nts, nte = ntf["coordinates"]["start"], ntf["coordinates"]["end"]
                        if not ntf["coordinates"]["lout"]:
                            p.lineTo(ntf["coordinates"]["start"], nty_c)
                            p.lineTo(ntf["coordinates"]["start"] + nt_arrow_len, nty_u)
                        else:
                            p.lineTo(ntf["coordinates"]["start"], nty_u)
                        p.lineTo(ntf["coordinates"]["end"], nty_u)
                        if nte <= cts:
                            p.lineTo(ntf["coordinates"]["end"], nty_c)
                    if (nte <= cts and ctf["coordinates"]["orient"] == 1) or (
                            nte >= cts and ctf["coordinates"]["orient"] == -1):
                        p.lineTo(cts, cty_c)
                    p.lineTo(cts, cty_b)
                    canvas.drawPath(p, stroke=1, fill=1)
    except Exception as error:
        raise lovis4u.Manager.lovis4uError("Unable to draw homology line track.") from error

LocusVis

Bases: Track

LocusVis track object that handles each locus visualisation.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class LocusVis(Track):
    """LocusVis track object that handles each locus visualisation.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        prms (lovis4u.Manager.Parameters): Parameters' class object.

    """

    def __init__(self, layout: dict, track_data: dict, parameters):
        """Create a LocusVis object.

        Arguments:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            track_data (dict): a dictionary with prepared track specific data.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.


        """
        super().__init__(layout, track_data, parameters)

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Draw a LocusVis track.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        try:
            y_track_bottom = self.layout["current_y_coordinate"] - self.track_data["track_height"]
            feature_height = self.prms.args["feature_height"] * mm

            y_feature_upper = self.layout["current_y_coordinate"] - (self.track_data["n_label_rows"] *
                                                                     self.track_data["f_label_height"] *
                                                                     (1 + self.prms.args["feature_label_gap"]))
            if self.prms.args["locus_label_position"] == "top_left":
                y_feature_upper -= self.prms.args["locus_label_height"] * 1.5

            if self.prms.args["locus_label_position"] == "top_center":
                y_feature_upper -= (self.prms.args["locus_label_height"] * 1.5 +
                                    self.track_data["ruler_label_height"] * 1.5)

            y_feature_bottom = y_feature_upper - feature_height
            y_feature_center = y_feature_upper - feature_height * 0.5

            # Sequence label
            canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("locus_label_colour", self.prms))
            canvas.setFont(self.prms.args["locus_label_description_font_face"], self.prms.args["locus_label_font_size"])
            if self.prms.args["locus_label_position"] == "left":
                if self.prms.args["locus_label_style"] == "full" and self.track_data["locus_description"]:
                    label_bottom = y_feature_upper - self.prms.args["locus_label_height"]
                    canvas.drawRightString(self.layout["locus_label_right_border"], label_bottom,
                                           self.track_data["locus_description"])
                    label_bottom = y_feature_bottom
                else:
                    label_bottom = y_feature_bottom + (feature_height - self.prms.args["locus_label_height"]) * 0.5
                    label = self.track_data["locus_description"]
                if self.prms.args["locus_label_style"] != "description":
                    label = self.track_data["locus_id"]
                    canvas.setFont(self.prms.args["locus_label_id_font_face"], self.prms.args["locus_label_font_size"])
                canvas.drawRightString(self.layout["locus_label_right_border"], label_bottom, label)
            else:
                if self.prms.args["locus_label_position"] == "bottom":
                    label_bottom = y_track_bottom
                    current_left = self.layout["locus_label_left_border"]
                elif self.prms.args["locus_label_position"] == "top_left":
                    label_bottom = self.layout["current_y_coordinate"] - self.prms.args["locus_label_height"]
                    current_left = self.prms.args["margin"] * mm
                elif self.prms.args["locus_label_position"] == "top_center":
                    label_bottom = self.layout["current_y_coordinate"] - self.prms.args["locus_label_height"]
                    total_label_width = self.track_data["locus_id_width"] + self.track_data["two_space_width"] + \
                                        self.track_data["text_coordinates_width"]
                    if self.prms.args["locus_label_style"] == "full" and self.track_data["locus_description"]:
                        total_label_width += self.track_data["two_space_width"] + self.track_data["locus_description_width"]
                    current_left = self.layout["figure_x_middle_position"] - total_label_width / 2

                if self.prms.args["locus_label_style"] == "full" and self.track_data["locus_description"]:
                    canvas.drawString(current_left, label_bottom, self.track_data["locus_description"])
                    current_left += self.track_data["locus_description_width"] + self.track_data["two_space_width"]
                canvas.setFont(self.prms.args["locus_label_id_font_face"], self.prms.args["locus_label_font_size"])
                canvas.drawString(current_left, label_bottom, self.track_data["locus_id"])
                current_left += self.track_data["locus_id_width"] + self.track_data["two_space_width"] /2
                canvas.drawString(current_left, label_bottom, self.track_data["text_coordinates"])

            if self.prms.args["locus_label_position"] == "top_center":
                ruler_label_bottom_position = self.layout["current_y_coordinate"] - \
                                       self.prms.args["locus_label_height"] * 1.5 - self.track_data["ruler_label_height"]
                ruler_line_middle = ruler_label_bottom_position + 0.5*self.track_data["ruler_label_height"]
                ruler_label_top_position = ruler_label_bottom_position + self.track_data["ruler_label_height"]
                ruler_line_height = self.track_data["ruler_label_height"]
                for region in self.track_data["coordinates_of_regions"]:
                    canvas.setLineWidth(self.prms.args["ruler_line_width"])
                    canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                    canvas.setLineCap(0)
                    canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                    region_middle = (region["start"] + region["end"])/2
                    left_border = region_middle - 0.5*(region["ruler_label_width"] + self.track_data["two_space_width"])
                    right_border = region_middle + 0.5*(region["ruler_label_width"] + self.track_data["two_space_width"])
                    canvas.line(region["start"], ruler_line_middle, left_border ,ruler_line_middle)
                    canvas.line(right_border, ruler_line_middle, region["end"] ,ruler_line_middle)
                    canvas.setLineCap(0)
                    canvas.line(region["start"], ruler_label_bottom_position, region["start"], ruler_label_top_position)
                    canvas.line(region["end"], ruler_label_bottom_position, region["end"], ruler_label_top_position)
                    canvas.setFont(self.prms.args["ruler_label_font_face"], self.prms.args["ruler_label_font_size"])
                    canvas.drawCentredString(region_middle, ruler_label_bottom_position, region["ruler_label"])



            # Middle line
            if self.prms.args["draw_middle_line"]:
                canvas.setLineWidth(self.prms.args["x_axis_line_width"])
                canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                canvas.setLineCap(0)
                canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                p = canvas.beginPath()
                for md_line_coordinates in self.track_data["middle_line_coordinates"]:
                    p.moveTo(md_line_coordinates["start"], y_feature_center)
                    p.lineTo(md_line_coordinates["end"], y_feature_center)
                canvas.drawPath(p, stroke=1, fill=0)

            # Category annotation
            if self.track_data["functions_coordinates"]:
                for feature_function, ff_region in self.track_data["functions_coordinates"].items():
                    feature_colour = self.track_data["category_colours"][feature_function]
                    canvas.setFillColorRGB(*matplotlib.colors.hex2color(feature_colour),
                                           self.prms.args["category_annotation_alpha"])
                    canvas.setLineJoin(1)
                    y_upper_sausage = y_feature_bottom - self.prms.args["feature_bottom_gap"] * mm
                    y_bottom_sausage = y_upper_sausage - self.prms.args["category_annotation_line_width"] * mm
                    for ffr in ff_region:
                        p = canvas.beginPath()
                        p.moveTo(ffr[0], y_bottom_sausage)
                        p.lineTo(ffr[0], y_upper_sausage)
                        p.lineTo(ffr[1], y_upper_sausage)
                        p.lineTo(ffr[1], y_bottom_sausage)
                        p.lineTo(ffr[0], y_bottom_sausage)
                        p.close()
                        canvas.drawPath(p, stroke=0, fill=1)
            # Features
            canvas.setLineCap(0)
            canvas.setLineJoin(1)
            canvas.setLineWidth(self.prms.args["feature_stroke_width"])
            if self.track_data["clean_features_coordinates"]:
                for f_data in self.track_data["features"]:
                    f_data_copy = copy.deepcopy(f_data)
                    f_data_copy["stroke_colour"] = None
                    f_data_copy["fill_colour"] = (*matplotlib.colors.hex2color(self.prms.args["palette"]["white"]), 1)
                    self.__plot_feature(canvas, f_data_copy, y_center=y_feature_center, height=feature_height)
            for f_data in self.track_data["features"]:
                f_data["stroke_colour"] = *matplotlib.colors.hex2color(f_data["stroke_colour"]), self.prms.args[
                    "feature_stroke_colour_alpha"]
                if f_data["fill_colour"]:
                    f_data["fill_colour"] = *matplotlib.colors.hex2color(f_data["fill_colour"]), self.prms.args[
                        "feature_fill_colour_alpha"]
                else:
                    f_data["fill_colour"] = None
                self.__plot_feature(canvas, f_data, y_center=y_feature_center, height=feature_height)
                if f_data["label_width"]:
                    canvas.setFillColorRGB(*f_data["stroke_colour"])
                    canvas.setFont(self.prms.args["feature_label_font_face"],
                                   self.track_data["f_label_font_size"])
                    fx_center = f_data["coordinates"]["center"]
                    canvas.drawString(f_data["label_position"][0], f_data["label_y_bottom"] + y_feature_upper,
                                      f_data["label"])
                    canvas.setLineWidth(self.prms.args["feature_stroke_width"])
                    underline_colour = f_data["stroke_colour"]
                    canvas.setStrokeColorRGB(*underline_colour)
                    canvas.setLineCap(1)
                    if f_data["label_row"] > 0:
                        p = canvas.beginPath()
                        for ls, le in f_data["label_line_coordinates"]:
                            p.moveTo(fx_center, ls + y_feature_upper)
                            p.lineTo(fx_center, le + y_feature_upper)
                        canvas.drawPath(p, stroke=1, fill=0)
                        l_start = f_data["coordinates"]["start"]
                        l_end = min(f_data["coordinates"]["end"], fx_center + self.track_data["feature_label_gap"])
                        l_end = f_data["coordinates"]["end"]
                        ly = y_feature_upper + f_data["label_y_bottom"] - self.track_data["feature_label_gap"] * 0.5
                        canvas.line(l_start, ly, l_end, ly)
                    else:
                        overlapping = min(f_data["coordinates"]["end"], f_data["label_position"][1]) - (
                            max(f_data["coordinates"]["start"], f_data["label_position"][0]))
                        if overlapping / (f_data["label_position"][1] - f_data["label_position"][0]) < 1:
                            l_start = max(f_data["coordinates"]["start"], fx_center -
                                          self.track_data["feature_label_gap"])
                            l_start = f_data["coordinates"]["start"]
                            l_end = min(f_data["coordinates"]["end"], fx_center + self.track_data["feature_label_gap"])
                            l_end = f_data["coordinates"]["end"]
                            ly = y_feature_upper + f_data["label_y_bottom"] - self.track_data["feature_label_gap"] * 0.5
                            canvas.line(l_start, ly, l_end, ly)
            # Axis ticks
            if self.prms.args["draw_individual_x_axis"]:
                canvas.setLineWidth(self.prms.args["x_axis_line_width"])
                canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                canvas.setLineCap(1)
                canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                canvas.setFont(self.prms.args["x_axis_ticks_labels_font_face"],
                               self.track_data["x_axis_annotation"]["label_size"])
                axis_line_y_coordinate = y_feature_bottom - self.prms.args["feature_bottom_gap"] * mm
                axis_tick_height = self.prms.args["x_axis_ticks_height"] * mm
                axis_tick_label_y_coordinate = axis_line_y_coordinate - self.prms.args["x_axis_ticks_height"] * \
                                               1.3 * mm - self.prms.args["x_axis_ticks_labels_height"] * mm
                for ati in range(len(self.track_data["x_axis_annotation"]["axis_tics_position"])):
                    tick_coordinate = self.track_data["x_axis_annotation"]["axis_tics_position"][ati]
                    tick_label_position = self.track_data["x_axis_annotation"]["tics_labels_coordinates"][ati]
                    tick_label = self.track_data["x_axis_annotation"]["axis_tics_labels"][ati]
                    canvas.drawCentredString(tick_label_position, axis_tick_label_y_coordinate, tick_label)
                    canvas.line(tick_coordinate, axis_line_y_coordinate, tick_coordinate,
                                axis_line_y_coordinate - axis_tick_height)
                for region in self.track_data["x_axis_annotation"]["axis_regions"]:
                    canvas.setLineCap(0)
                    canvas.line(region["start"], axis_line_y_coordinate, region["end"], axis_line_y_coordinate)
            return None
        except Exception as error:
            raise lovis4u.Manager.lovis4uError("Unable to draw a Locus track.") from error

    def __plot_feature(self, canvas: reportlab.pdfgen.canvas.Canvas, feature_data: dict, y_center: float,
                       height: float) -> None:
        """Helper method to plot feature polygone

        Returns:
            None

        """
        x_start = feature_data["coordinates"]["start"]
        x_end = feature_data["coordinates"]["end"]
        orientation = feature_data["coordinates"]["orient"]
        left_out = feature_data["coordinates"]["lout"]
        right_out = feature_data["coordinates"]["rout"]
        fill_colour = feature_data["fill_colour"]
        stroke_colour = feature_data["stroke_colour"]
        feature_type = feature_data["type"]
        y_center = y_center
        height = height

        canvas.setLineCap(0)
        canvas.setLineWidth(self.prms.args["feature_stroke_width"])
        arrow_length = min(height * self.prms.args["feature_arrow_length"], (x_end - x_start))
        p_noncoding = canvas.beginPath()
        if feature_type != "CDS":
            p_noncoding.moveTo(x_start, y_center + height / 2)
            p_noncoding.lineTo(x_end, y_center + height / 2)
            if not right_out:
                p_noncoding.lineTo(x_end, y_center - height / 2)
            else:
                p_noncoding.moveTo(x_end, y_center - height / 2)
            p_noncoding.lineTo(x_start, y_center - height / 2)
            if not left_out:
                p_noncoding.lineTo(x_start, y_center + height / 2)
            if not left_out and not right_out:
                p_noncoding.close()
            stroke, fill = 0, 0
            if stroke_colour:
                canvas.setStrokeColorRGB(*stroke_colour)
                stroke = 1
            if fill_colour:
                canvas.setFillColorRGB(*fill_colour)
                fill = 1
            canvas.drawPath(p_noncoding, stroke=stroke, fill=fill)
        p = canvas.beginPath()
        if orientation == 1:
            if right_out:
                p.moveTo(x_end, y_center - height / 2)
                p.lineTo(x_start, y_center - height / 2)
                p.lineTo(x_start, y_center + height / 2)
                p.lineTo(x_end, y_center + height / 2)
            else:
                p.moveTo(x_start, y_center + height / 2)
                p.lineTo(x_end - arrow_length, y_center + height / 2)
                p.lineTo(x_end, y_center)
                p.lineTo(x_end - arrow_length, y_center - height / 2)
                p.lineTo(x_start, y_center - height / 2)
                if not left_out:
                    p.lineTo(x_start, y_center + height / 2)
        elif orientation == -1:
            if left_out:
                p.moveTo(x_start, y_center - height / 2)
                p.lineTo(x_end, y_center - height / 2)
                p.lineTo(x_end, y_center + height / 2)
                p.lineTo(x_start, y_center + height / 2)
            else:
                p.moveTo(x_end, y_center + height / 2)
                p.lineTo(x_start + arrow_length, y_center + height / 2)
                p.lineTo(x_start, y_center)
                p.lineTo(x_start + arrow_length, y_center - height / 2)
                p.lineTo(x_end, y_center - height / 2)
                if not right_out:
                    p.lineTo(x_end, y_center + height / 2)
        if left_out and right_out:
            p.moveTo(x_start, y_center + height / 2)
            p.lineTo(x_end, y_center + height / 2)
            p.moveTo(x_start, y_center - height / 2)
            p.lineTo(x_end, y_center - height / 2)
        if not left_out and not right_out:
            p.close()
        stroke, fill = 0, 0
        if stroke_colour:
            canvas.setStrokeColorRGB(*stroke_colour)
            stroke = 1
        if fill_colour:
            canvas.setFillColorRGB(*fill_colour)
            fill = 1
        canvas.drawPath(p, stroke=stroke, fill=fill)
        return None

__init__(layout, track_data, parameters)

Create a LocusVis object.

Parameters:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout: dict, track_data: dict, parameters):
    """Create a LocusVis object.

    Arguments:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.


    """
    super().__init__(layout, track_data, parameters)

__plot_feature(canvas, feature_data, y_center, height)

Helper method to plot feature polygone

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def __plot_feature(self, canvas: reportlab.pdfgen.canvas.Canvas, feature_data: dict, y_center: float,
                   height: float) -> None:
    """Helper method to plot feature polygone

    Returns:
        None

    """
    x_start = feature_data["coordinates"]["start"]
    x_end = feature_data["coordinates"]["end"]
    orientation = feature_data["coordinates"]["orient"]
    left_out = feature_data["coordinates"]["lout"]
    right_out = feature_data["coordinates"]["rout"]
    fill_colour = feature_data["fill_colour"]
    stroke_colour = feature_data["stroke_colour"]
    feature_type = feature_data["type"]
    y_center = y_center
    height = height

    canvas.setLineCap(0)
    canvas.setLineWidth(self.prms.args["feature_stroke_width"])
    arrow_length = min(height * self.prms.args["feature_arrow_length"], (x_end - x_start))
    p_noncoding = canvas.beginPath()
    if feature_type != "CDS":
        p_noncoding.moveTo(x_start, y_center + height / 2)
        p_noncoding.lineTo(x_end, y_center + height / 2)
        if not right_out:
            p_noncoding.lineTo(x_end, y_center - height / 2)
        else:
            p_noncoding.moveTo(x_end, y_center - height / 2)
        p_noncoding.lineTo(x_start, y_center - height / 2)
        if not left_out:
            p_noncoding.lineTo(x_start, y_center + height / 2)
        if not left_out and not right_out:
            p_noncoding.close()
        stroke, fill = 0, 0
        if stroke_colour:
            canvas.setStrokeColorRGB(*stroke_colour)
            stroke = 1
        if fill_colour:
            canvas.setFillColorRGB(*fill_colour)
            fill = 1
        canvas.drawPath(p_noncoding, stroke=stroke, fill=fill)
    p = canvas.beginPath()
    if orientation == 1:
        if right_out:
            p.moveTo(x_end, y_center - height / 2)
            p.lineTo(x_start, y_center - height / 2)
            p.lineTo(x_start, y_center + height / 2)
            p.lineTo(x_end, y_center + height / 2)
        else:
            p.moveTo(x_start, y_center + height / 2)
            p.lineTo(x_end - arrow_length, y_center + height / 2)
            p.lineTo(x_end, y_center)
            p.lineTo(x_end - arrow_length, y_center - height / 2)
            p.lineTo(x_start, y_center - height / 2)
            if not left_out:
                p.lineTo(x_start, y_center + height / 2)
    elif orientation == -1:
        if left_out:
            p.moveTo(x_start, y_center - height / 2)
            p.lineTo(x_end, y_center - height / 2)
            p.lineTo(x_end, y_center + height / 2)
            p.lineTo(x_start, y_center + height / 2)
        else:
            p.moveTo(x_end, y_center + height / 2)
            p.lineTo(x_start + arrow_length, y_center + height / 2)
            p.lineTo(x_start, y_center)
            p.lineTo(x_start + arrow_length, y_center - height / 2)
            p.lineTo(x_end, y_center - height / 2)
            if not right_out:
                p.lineTo(x_end, y_center + height / 2)
    if left_out and right_out:
        p.moveTo(x_start, y_center + height / 2)
        p.lineTo(x_end, y_center + height / 2)
        p.moveTo(x_start, y_center - height / 2)
        p.lineTo(x_end, y_center - height / 2)
    if not left_out and not right_out:
        p.close()
    stroke, fill = 0, 0
    if stroke_colour:
        canvas.setStrokeColorRGB(*stroke_colour)
        stroke = 1
    if fill_colour:
        canvas.setFillColorRGB(*fill_colour)
        fill = 1
    canvas.drawPath(p, stroke=stroke, fill=fill)
    return None

draw(canvas)

Draw a LocusVis track.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Draw a LocusVis track.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    try:
        y_track_bottom = self.layout["current_y_coordinate"] - self.track_data["track_height"]
        feature_height = self.prms.args["feature_height"] * mm

        y_feature_upper = self.layout["current_y_coordinate"] - (self.track_data["n_label_rows"] *
                                                                 self.track_data["f_label_height"] *
                                                                 (1 + self.prms.args["feature_label_gap"]))
        if self.prms.args["locus_label_position"] == "top_left":
            y_feature_upper -= self.prms.args["locus_label_height"] * 1.5

        if self.prms.args["locus_label_position"] == "top_center":
            y_feature_upper -= (self.prms.args["locus_label_height"] * 1.5 +
                                self.track_data["ruler_label_height"] * 1.5)

        y_feature_bottom = y_feature_upper - feature_height
        y_feature_center = y_feature_upper - feature_height * 0.5

        # Sequence label
        canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("locus_label_colour", self.prms))
        canvas.setFont(self.prms.args["locus_label_description_font_face"], self.prms.args["locus_label_font_size"])
        if self.prms.args["locus_label_position"] == "left":
            if self.prms.args["locus_label_style"] == "full" and self.track_data["locus_description"]:
                label_bottom = y_feature_upper - self.prms.args["locus_label_height"]
                canvas.drawRightString(self.layout["locus_label_right_border"], label_bottom,
                                       self.track_data["locus_description"])
                label_bottom = y_feature_bottom
            else:
                label_bottom = y_feature_bottom + (feature_height - self.prms.args["locus_label_height"]) * 0.5
                label = self.track_data["locus_description"]
            if self.prms.args["locus_label_style"] != "description":
                label = self.track_data["locus_id"]
                canvas.setFont(self.prms.args["locus_label_id_font_face"], self.prms.args["locus_label_font_size"])
            canvas.drawRightString(self.layout["locus_label_right_border"], label_bottom, label)
        else:
            if self.prms.args["locus_label_position"] == "bottom":
                label_bottom = y_track_bottom
                current_left = self.layout["locus_label_left_border"]
            elif self.prms.args["locus_label_position"] == "top_left":
                label_bottom = self.layout["current_y_coordinate"] - self.prms.args["locus_label_height"]
                current_left = self.prms.args["margin"] * mm
            elif self.prms.args["locus_label_position"] == "top_center":
                label_bottom = self.layout["current_y_coordinate"] - self.prms.args["locus_label_height"]
                total_label_width = self.track_data["locus_id_width"] + self.track_data["two_space_width"] + \
                                    self.track_data["text_coordinates_width"]
                if self.prms.args["locus_label_style"] == "full" and self.track_data["locus_description"]:
                    total_label_width += self.track_data["two_space_width"] + self.track_data["locus_description_width"]
                current_left = self.layout["figure_x_middle_position"] - total_label_width / 2

            if self.prms.args["locus_label_style"] == "full" and self.track_data["locus_description"]:
                canvas.drawString(current_left, label_bottom, self.track_data["locus_description"])
                current_left += self.track_data["locus_description_width"] + self.track_data["two_space_width"]
            canvas.setFont(self.prms.args["locus_label_id_font_face"], self.prms.args["locus_label_font_size"])
            canvas.drawString(current_left, label_bottom, self.track_data["locus_id"])
            current_left += self.track_data["locus_id_width"] + self.track_data["two_space_width"] /2
            canvas.drawString(current_left, label_bottom, self.track_data["text_coordinates"])

        if self.prms.args["locus_label_position"] == "top_center":
            ruler_label_bottom_position = self.layout["current_y_coordinate"] - \
                                   self.prms.args["locus_label_height"] * 1.5 - self.track_data["ruler_label_height"]
            ruler_line_middle = ruler_label_bottom_position + 0.5*self.track_data["ruler_label_height"]
            ruler_label_top_position = ruler_label_bottom_position + self.track_data["ruler_label_height"]
            ruler_line_height = self.track_data["ruler_label_height"]
            for region in self.track_data["coordinates_of_regions"]:
                canvas.setLineWidth(self.prms.args["ruler_line_width"])
                canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                canvas.setLineCap(0)
                canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                region_middle = (region["start"] + region["end"])/2
                left_border = region_middle - 0.5*(region["ruler_label_width"] + self.track_data["two_space_width"])
                right_border = region_middle + 0.5*(region["ruler_label_width"] + self.track_data["two_space_width"])
                canvas.line(region["start"], ruler_line_middle, left_border ,ruler_line_middle)
                canvas.line(right_border, ruler_line_middle, region["end"] ,ruler_line_middle)
                canvas.setLineCap(0)
                canvas.line(region["start"], ruler_label_bottom_position, region["start"], ruler_label_top_position)
                canvas.line(region["end"], ruler_label_bottom_position, region["end"], ruler_label_top_position)
                canvas.setFont(self.prms.args["ruler_label_font_face"], self.prms.args["ruler_label_font_size"])
                canvas.drawCentredString(region_middle, ruler_label_bottom_position, region["ruler_label"])



        # Middle line
        if self.prms.args["draw_middle_line"]:
            canvas.setLineWidth(self.prms.args["x_axis_line_width"])
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setLineCap(0)
            canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            p = canvas.beginPath()
            for md_line_coordinates in self.track_data["middle_line_coordinates"]:
                p.moveTo(md_line_coordinates["start"], y_feature_center)
                p.lineTo(md_line_coordinates["end"], y_feature_center)
            canvas.drawPath(p, stroke=1, fill=0)

        # Category annotation
        if self.track_data["functions_coordinates"]:
            for feature_function, ff_region in self.track_data["functions_coordinates"].items():
                feature_colour = self.track_data["category_colours"][feature_function]
                canvas.setFillColorRGB(*matplotlib.colors.hex2color(feature_colour),
                                       self.prms.args["category_annotation_alpha"])
                canvas.setLineJoin(1)
                y_upper_sausage = y_feature_bottom - self.prms.args["feature_bottom_gap"] * mm
                y_bottom_sausage = y_upper_sausage - self.prms.args["category_annotation_line_width"] * mm
                for ffr in ff_region:
                    p = canvas.beginPath()
                    p.moveTo(ffr[0], y_bottom_sausage)
                    p.lineTo(ffr[0], y_upper_sausage)
                    p.lineTo(ffr[1], y_upper_sausage)
                    p.lineTo(ffr[1], y_bottom_sausage)
                    p.lineTo(ffr[0], y_bottom_sausage)
                    p.close()
                    canvas.drawPath(p, stroke=0, fill=1)
        # Features
        canvas.setLineCap(0)
        canvas.setLineJoin(1)
        canvas.setLineWidth(self.prms.args["feature_stroke_width"])
        if self.track_data["clean_features_coordinates"]:
            for f_data in self.track_data["features"]:
                f_data_copy = copy.deepcopy(f_data)
                f_data_copy["stroke_colour"] = None
                f_data_copy["fill_colour"] = (*matplotlib.colors.hex2color(self.prms.args["palette"]["white"]), 1)
                self.__plot_feature(canvas, f_data_copy, y_center=y_feature_center, height=feature_height)
        for f_data in self.track_data["features"]:
            f_data["stroke_colour"] = *matplotlib.colors.hex2color(f_data["stroke_colour"]), self.prms.args[
                "feature_stroke_colour_alpha"]
            if f_data["fill_colour"]:
                f_data["fill_colour"] = *matplotlib.colors.hex2color(f_data["fill_colour"]), self.prms.args[
                    "feature_fill_colour_alpha"]
            else:
                f_data["fill_colour"] = None
            self.__plot_feature(canvas, f_data, y_center=y_feature_center, height=feature_height)
            if f_data["label_width"]:
                canvas.setFillColorRGB(*f_data["stroke_colour"])
                canvas.setFont(self.prms.args["feature_label_font_face"],
                               self.track_data["f_label_font_size"])
                fx_center = f_data["coordinates"]["center"]
                canvas.drawString(f_data["label_position"][0], f_data["label_y_bottom"] + y_feature_upper,
                                  f_data["label"])
                canvas.setLineWidth(self.prms.args["feature_stroke_width"])
                underline_colour = f_data["stroke_colour"]
                canvas.setStrokeColorRGB(*underline_colour)
                canvas.setLineCap(1)
                if f_data["label_row"] > 0:
                    p = canvas.beginPath()
                    for ls, le in f_data["label_line_coordinates"]:
                        p.moveTo(fx_center, ls + y_feature_upper)
                        p.lineTo(fx_center, le + y_feature_upper)
                    canvas.drawPath(p, stroke=1, fill=0)
                    l_start = f_data["coordinates"]["start"]
                    l_end = min(f_data["coordinates"]["end"], fx_center + self.track_data["feature_label_gap"])
                    l_end = f_data["coordinates"]["end"]
                    ly = y_feature_upper + f_data["label_y_bottom"] - self.track_data["feature_label_gap"] * 0.5
                    canvas.line(l_start, ly, l_end, ly)
                else:
                    overlapping = min(f_data["coordinates"]["end"], f_data["label_position"][1]) - (
                        max(f_data["coordinates"]["start"], f_data["label_position"][0]))
                    if overlapping / (f_data["label_position"][1] - f_data["label_position"][0]) < 1:
                        l_start = max(f_data["coordinates"]["start"], fx_center -
                                      self.track_data["feature_label_gap"])
                        l_start = f_data["coordinates"]["start"]
                        l_end = min(f_data["coordinates"]["end"], fx_center + self.track_data["feature_label_gap"])
                        l_end = f_data["coordinates"]["end"]
                        ly = y_feature_upper + f_data["label_y_bottom"] - self.track_data["feature_label_gap"] * 0.5
                        canvas.line(l_start, ly, l_end, ly)
        # Axis ticks
        if self.prms.args["draw_individual_x_axis"]:
            canvas.setLineWidth(self.prms.args["x_axis_line_width"])
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setLineCap(1)
            canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setFont(self.prms.args["x_axis_ticks_labels_font_face"],
                           self.track_data["x_axis_annotation"]["label_size"])
            axis_line_y_coordinate = y_feature_bottom - self.prms.args["feature_bottom_gap"] * mm
            axis_tick_height = self.prms.args["x_axis_ticks_height"] * mm
            axis_tick_label_y_coordinate = axis_line_y_coordinate - self.prms.args["x_axis_ticks_height"] * \
                                           1.3 * mm - self.prms.args["x_axis_ticks_labels_height"] * mm
            for ati in range(len(self.track_data["x_axis_annotation"]["axis_tics_position"])):
                tick_coordinate = self.track_data["x_axis_annotation"]["axis_tics_position"][ati]
                tick_label_position = self.track_data["x_axis_annotation"]["tics_labels_coordinates"][ati]
                tick_label = self.track_data["x_axis_annotation"]["axis_tics_labels"][ati]
                canvas.drawCentredString(tick_label_position, axis_tick_label_y_coordinate, tick_label)
                canvas.line(tick_coordinate, axis_line_y_coordinate, tick_coordinate,
                            axis_line_y_coordinate - axis_tick_height)
            for region in self.track_data["x_axis_annotation"]["axis_regions"]:
                canvas.setLineCap(0)
                canvas.line(region["start"], axis_line_y_coordinate, region["end"], axis_line_y_coordinate)
        return None
    except Exception as error:
        raise lovis4u.Manager.lovis4uError("Unable to draw a Locus track.") from error

ProfileVis

Bases: Track

ProfileVis track object that handles visualisation of a coverage profile.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class ProfileVis(Track):
    """ProfileVis track object that handles visualisation of a coverage profile.

       Attributes:
           layout (dict): Layout built by CanvasManager's define_layout() method.
           track_data (dict): a dictionary with prepared track specific data.
           prms (lovis4u.Manager.Parameters): Parameters' class object.

       """

    def __init__(self, layout: dict, track_data: dict, parameters):
        """Create a ProfileVis object.

        Arguments:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            track_data (dict): a dictionary with prepared track specific data.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.

        """
        super().__init__(layout, track_data, parameters)

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Draw a ProfileVis track.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        try:
            y_upper = self.layout["current_y_coordinate"]
            y_bottom = y_upper - self.track_data["track_height"]

            if self.prms.args["bedgraph_profile_style"] == "bins":
                canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["colour"]),
                                       self.prms.args["bedgraph_track_colour_alpha"])
                for vis_region in self.track_data["vis_regions"]:
                    canvas.line(vis_region["canvas_start"], y_bottom, vis_region["canvas_end"], y_bottom)
                    current_x = vis_region["canvas_start"]
                    for bin in vis_region["region_coverage"]:
                        canvas.rect(current_x + 0.01 * vis_region["bin_width"], y_bottom,
                                    vis_region["bin_width"] * 0.98,
                                    bin * self.track_data["y_scale"], fill=1, stroke=0)
                        current_x += vis_region["bin_width"]

            elif self.prms.args["bedgraph_profile_style"] == "line":
                canvas.setDash(1)
                canvas.setLineCap(0)
                canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["colour"]),
                                       self.prms.args["bedgraph_track_colour_alpha"])
                canvas.setLineWidth(self.prms.args["bedgraph_track_bottom_line_width"])
                for vis_region in self.track_data["vis_regions"]:
                    current_x = vis_region["canvas_start"]
                    path_fill, path_stroke = canvas.beginPath(), canvas.beginPath()
                    path_fill.moveTo(vis_region["canvas_start"], y_bottom)
                    for bin_index, bin in enumerate(vis_region["region_coverage"]):
                        path_fill.lineTo(current_x, y_bottom + bin * self.track_data["y_scale"])
                        path_fill.lineTo(current_x + vis_region["bin_width"],
                                         y_bottom + bin * self.track_data["y_scale"])
                        if bin_index == 0:
                            path_stroke.moveTo(current_x, y_bottom + bin * self.track_data["y_scale"])
                        path_stroke.lineTo(current_x, y_bottom + bin * self.track_data["y_scale"])
                        path_stroke.lineTo(current_x + vis_region["bin_width"],
                                           y_bottom + bin * self.track_data["y_scale"])
                        current_x += vis_region["bin_width"]
                    path_fill.lineTo(current_x, y_bottom)
                    path_fill.lineTo(vis_region["canvas_start"], y_bottom)
                    path_fill.close()
                    canvas.drawPath(path_fill, stroke=0, fill=1)
                    canvas.setStrokeColorRGB(*matplotlib.colors.hex2color(
                        lovis4u.Methods.scale_lightness(self.track_data["colour"], 0.8)), 1)
                    canvas.drawPath(path_stroke, stroke=1, fill=0)

            for vis_region in self.track_data["vis_regions"]:
                canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                canvas.setLineCap(0)
                canvas.setDash(1)
                canvas.setLineWidth(self.prms.args["bedgraph_track_bottom_line_width"])
                canvas.line(vis_region["canvas_start"], y_bottom, vis_region["canvas_end"], y_bottom)
                canvas.setDash(self.prms.args["bedgraph_track_bottom_line_width"] / 2, 1)
                canvas.setLineWidth(self.prms.args["bedgraph_track_bottom_line_width"] / 2)
                canvas.line(vis_region["canvas_start"], y_upper, vis_region["canvas_end"], y_upper)

            # Profile Label
            canvas.setFont(self.prms.args["bedgraph_label_font_face"], self.prms.args["bedgraph_label_font_size"])
            label_y_bottom = y_bottom + 0.3 * self.track_data["label_height"]
            canvas.setFillColorRGB(
                *matplotlib.colors.hex2color(lovis4u.Methods.scale_lightness(self.track_data["colour"], 0.25)), 1)
            # canvas.drawCentredString(self.layout["figure_x_middle_position"], label_y_bottom, self.track_data["label_text"])
            canvas.drawString(self.layout["loci_tracks_left_border"], label_y_bottom, self.track_data["label_text"])

            # Axis annotation
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setLineWidth(self.prms.args["property_track_middle_line_width"])
            canvas.setDash(1)
            canvas.setLineCap(0)
            left_border = self.layout["loci_tracks_left_border"]
            right_border = self.layout["loci_tracks_right_border"]
            axis_label_height = self.track_data["axis_labels_height"]
            canvas.setFont(self.prms.args["property_axis_font_face"], self.prms.args["property_axis_font_size"])
            full_bottom_annotation_text = f"[{round(self.track_data['y_min'], 2)}-{round(self.track_data['y_max'], 2)}]"
            canvas.drawRightString(right_border, y_bottom + 0.3 * axis_label_height, full_bottom_annotation_text)

            return None
        except Exception as error:
            raise lovis4u.Manager.lovis4uError("Unable to draw a profile coverage track.") from error

__init__(layout, track_data, parameters)

Create a ProfileVis object.

Parameters:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout: dict, track_data: dict, parameters):
    """Create a ProfileVis object.

    Arguments:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.

    """
    super().__init__(layout, track_data, parameters)

draw(canvas)

Draw a ProfileVis track.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Draw a ProfileVis track.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    try:
        y_upper = self.layout["current_y_coordinate"]
        y_bottom = y_upper - self.track_data["track_height"]

        if self.prms.args["bedgraph_profile_style"] == "bins":
            canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["colour"]),
                                   self.prms.args["bedgraph_track_colour_alpha"])
            for vis_region in self.track_data["vis_regions"]:
                canvas.line(vis_region["canvas_start"], y_bottom, vis_region["canvas_end"], y_bottom)
                current_x = vis_region["canvas_start"]
                for bin in vis_region["region_coverage"]:
                    canvas.rect(current_x + 0.01 * vis_region["bin_width"], y_bottom,
                                vis_region["bin_width"] * 0.98,
                                bin * self.track_data["y_scale"], fill=1, stroke=0)
                    current_x += vis_region["bin_width"]

        elif self.prms.args["bedgraph_profile_style"] == "line":
            canvas.setDash(1)
            canvas.setLineCap(0)
            canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["colour"]),
                                   self.prms.args["bedgraph_track_colour_alpha"])
            canvas.setLineWidth(self.prms.args["bedgraph_track_bottom_line_width"])
            for vis_region in self.track_data["vis_regions"]:
                current_x = vis_region["canvas_start"]
                path_fill, path_stroke = canvas.beginPath(), canvas.beginPath()
                path_fill.moveTo(vis_region["canvas_start"], y_bottom)
                for bin_index, bin in enumerate(vis_region["region_coverage"]):
                    path_fill.lineTo(current_x, y_bottom + bin * self.track_data["y_scale"])
                    path_fill.lineTo(current_x + vis_region["bin_width"],
                                     y_bottom + bin * self.track_data["y_scale"])
                    if bin_index == 0:
                        path_stroke.moveTo(current_x, y_bottom + bin * self.track_data["y_scale"])
                    path_stroke.lineTo(current_x, y_bottom + bin * self.track_data["y_scale"])
                    path_stroke.lineTo(current_x + vis_region["bin_width"],
                                       y_bottom + bin * self.track_data["y_scale"])
                    current_x += vis_region["bin_width"]
                path_fill.lineTo(current_x, y_bottom)
                path_fill.lineTo(vis_region["canvas_start"], y_bottom)
                path_fill.close()
                canvas.drawPath(path_fill, stroke=0, fill=1)
                canvas.setStrokeColorRGB(*matplotlib.colors.hex2color(
                    lovis4u.Methods.scale_lightness(self.track_data["colour"], 0.8)), 1)
                canvas.drawPath(path_stroke, stroke=1, fill=0)

        for vis_region in self.track_data["vis_regions"]:
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setLineCap(0)
            canvas.setDash(1)
            canvas.setLineWidth(self.prms.args["bedgraph_track_bottom_line_width"])
            canvas.line(vis_region["canvas_start"], y_bottom, vis_region["canvas_end"], y_bottom)
            canvas.setDash(self.prms.args["bedgraph_track_bottom_line_width"] / 2, 1)
            canvas.setLineWidth(self.prms.args["bedgraph_track_bottom_line_width"] / 2)
            canvas.line(vis_region["canvas_start"], y_upper, vis_region["canvas_end"], y_upper)

        # Profile Label
        canvas.setFont(self.prms.args["bedgraph_label_font_face"], self.prms.args["bedgraph_label_font_size"])
        label_y_bottom = y_bottom + 0.3 * self.track_data["label_height"]
        canvas.setFillColorRGB(
            *matplotlib.colors.hex2color(lovis4u.Methods.scale_lightness(self.track_data["colour"], 0.25)), 1)
        # canvas.drawCentredString(self.layout["figure_x_middle_position"], label_y_bottom, self.track_data["label_text"])
        canvas.drawString(self.layout["loci_tracks_left_border"], label_y_bottom, self.track_data["label_text"])

        # Axis annotation
        canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
        canvas.setLineWidth(self.prms.args["property_track_middle_line_width"])
        canvas.setDash(1)
        canvas.setLineCap(0)
        left_border = self.layout["loci_tracks_left_border"]
        right_border = self.layout["loci_tracks_right_border"]
        axis_label_height = self.track_data["axis_labels_height"]
        canvas.setFont(self.prms.args["property_axis_font_face"], self.prms.args["property_axis_font_size"])
        full_bottom_annotation_text = f"[{round(self.track_data['y_min'], 2)}-{round(self.track_data['y_max'], 2)}]"
        canvas.drawRightString(right_border, y_bottom + 0.3 * axis_label_height, full_bottom_annotation_text)

        return None
    except Exception as error:
        raise lovis4u.Manager.lovis4uError("Unable to draw a profile coverage track.") from error

PropertyVis

Bases: Track

PropertyVis track object that handles visualisation of a coverage profile.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class PropertyVis(Track):
    """PropertyVis track object that handles visualisation of a coverage profile.

       Attributes:
           layout (dict): Layout built by CanvasManager's define_layout() method.
           track_data (dict): a dictionary with prepared track specific data.
           prms (lovis4u.Manager.Parameters): Parameters' class object.

       """

    def __init__(self, layout: dict, track_data: dict, parameters):
        """Create a PropertyVis object.

        Arguments:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            track_data (dict): a dictionary with prepared track specific data.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.

        """
        super().__init__(layout, track_data, parameters)

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Draw a PropertyVis track.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        try:
            y_upper = self.layout["current_y_coordinate"]
            y_bottom = y_upper - self.track_data["track_height"]
            y_middle = (y_upper + y_bottom) / 2

            canvas.setDash(1)
            canvas.setLineCap(0)
            canvas.setLineWidth(self.prms.args["property_track_stroke_line_width"])
            for vis_region in self.track_data["vis_regions"]:
                current_x = vis_region["canvas_start"]
                path_stroke_positive, path_stroke_negative = canvas.beginPath(), canvas.beginPath()
                path_fill_positive, path_fill_negative = canvas.beginPath(), canvas.beginPath()
                path_fill_positive.moveTo(vis_region["canvas_start"], y_middle)
                path_fill_negative.moveTo(vis_region["canvas_start"], y_middle)
                previous_state = "NA"
                previous_fill_path, previous_stroke_path = None, None
                for bin_index, bin_value in enumerate(vis_region["region_values"]):
                    bin_height = (bin_value - self.track_data["y_min"]) * self.track_data["y_scale"]
                    current_state = "positive" if bin_value >= self.track_data["y_average"] else "negative"
                    current_stroke_path = path_stroke_positive if current_state == "positive" else path_stroke_negative
                    current_fill_path = path_fill_positive if current_state == "positive" else path_fill_negative
                    if current_state != previous_state:
                        if previous_fill_path:
                            previous_fill_path.lineTo(current_x - vis_region["bin_width"], y_middle)
                            previous_stroke_path.lineTo(current_x - vis_region["bin_width"], y_middle)
                        current_fill_path.lineTo(current_x, y_middle)
                        current_stroke_path.moveTo(current_x, y_middle)
                    current_fill_path.lineTo(current_x, y_bottom + bin_height)
                    current_fill_path.lineTo(current_x + vis_region["bin_width"], y_bottom + bin_height)
                    current_stroke_path.lineTo(current_x, y_bottom + bin_height)
                    current_stroke_path.lineTo(current_x + vis_region["bin_width"], y_bottom + bin_height)
                    current_x += vis_region["bin_width"]
                    previous_state = current_state
                    previous_fill_path = current_fill_path
                    previous_stroke_path = current_stroke_path
                canvas.setStrokeColorRGB(*matplotlib.colors.hex2color(
                    lovis4u.Methods.scale_lightness(self.track_data["positive_colour"], 0.8)), 1)
                canvas.drawPath(path_stroke_positive, stroke=1, fill=0)
                canvas.setStrokeColorRGB(*matplotlib.colors.hex2color(
                    lovis4u.Methods.scale_lightness(self.track_data["negative_colour"], 0.8)), 1)
                canvas.drawPath(path_stroke_negative, stroke=1, fill=0)

                path_fill_positive.lineTo(vis_region["canvas_end"], y_middle)
                path_fill_negative.lineTo(vis_region["canvas_end"], y_middle)
                path_fill_positive.lineTo(vis_region["canvas_start"], y_middle)
                path_fill_negative.lineTo(vis_region["canvas_start"], y_middle)
                path_fill_positive.close()
                path_fill_negative.close()
                canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["positive_colour"]),
                                       self.prms.args["property_track_colour_alpha"])
                canvas.drawPath(path_fill_positive, stroke=0, fill=1)
                canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["negative_colour"]),
                                       self.prms.args["property_track_colour_alpha"])
                canvas.drawPath(path_fill_negative, stroke=0, fill=1)

            for vis_region in self.track_data["vis_regions"]:
                canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
                canvas.setLineCap(0)
                canvas.setLineWidth(self.prms.args["property_track_middle_line_width"])
                canvas.setDash(1)
                canvas.line(vis_region["canvas_start"], (y_bottom + y_upper) / 2, vis_region["canvas_end"],
                            (y_bottom + y_upper) / 2)
                canvas.setDash(self.prms.args["property_track_middle_line_width"] / 2, 1)
                canvas.setLineWidth(self.prms.args["property_track_middle_line_width"] / 2)
                canvas.line(vis_region["canvas_start"], y_upper, vis_region["canvas_end"], y_upper)
                canvas.line(vis_region["canvas_start"], y_bottom, vis_region["canvas_end"], y_bottom)

            # Profile Label
            canvas.setFont(self.prms.args["property_label_font_face"], self.prms.args["property_label_font_size"])
            label_y_bottom = y_bottom + 0.3 * self.track_data["label_height"]
            canvas.setFillColorRGB(
                *matplotlib.colors.hex2color(lovis4u.Methods.scale_lightness(self.track_data["positive_colour"], 0.25)),
                1)
            # canvas.drawCentredString(self.layout["figure_x_middle_position"], label_y_bottom, self.track_data["label_text"])
            canvas.drawString(self.layout["loci_tracks_left_border"], label_y_bottom, self.track_data["label_text"])

            # Axis annotation
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setLineWidth(self.prms.args["property_track_middle_line_width"])
            canvas.setDash(1)
            canvas.setLineCap(0)
            left_border = self.layout["loci_tracks_left_border"]
            right_border = self.layout["loci_tracks_right_border"]
            axis_label_height = self.track_data["axis_labels_height"]
            canvas.setFont(self.prms.args["property_axis_font_face"], self.prms.args["property_axis_font_size"])
            full_bottom_annotation_text = f"Average: {round(self.track_data['y_average'], 2)}; Range: [" \
                                          f"{round(self.track_data['y_min'], 2)}, {round(self.track_data['y_max'], 2)}]"
            canvas.drawRightString(right_border, y_bottom + 0.3 * axis_label_height, full_bottom_annotation_text)

            return None
        except Exception as error:
            raise lovis4u.Manager.lovis4uError("Unable to draw a property track.") from error

__init__(layout, track_data, parameters)

Create a PropertyVis object.

Parameters:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout: dict, track_data: dict, parameters):
    """Create a PropertyVis object.

    Arguments:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.

    """
    super().__init__(layout, track_data, parameters)

draw(canvas)

Draw a PropertyVis track.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Draw a PropertyVis track.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    try:
        y_upper = self.layout["current_y_coordinate"]
        y_bottom = y_upper - self.track_data["track_height"]
        y_middle = (y_upper + y_bottom) / 2

        canvas.setDash(1)
        canvas.setLineCap(0)
        canvas.setLineWidth(self.prms.args["property_track_stroke_line_width"])
        for vis_region in self.track_data["vis_regions"]:
            current_x = vis_region["canvas_start"]
            path_stroke_positive, path_stroke_negative = canvas.beginPath(), canvas.beginPath()
            path_fill_positive, path_fill_negative = canvas.beginPath(), canvas.beginPath()
            path_fill_positive.moveTo(vis_region["canvas_start"], y_middle)
            path_fill_negative.moveTo(vis_region["canvas_start"], y_middle)
            previous_state = "NA"
            previous_fill_path, previous_stroke_path = None, None
            for bin_index, bin_value in enumerate(vis_region["region_values"]):
                bin_height = (bin_value - self.track_data["y_min"]) * self.track_data["y_scale"]
                current_state = "positive" if bin_value >= self.track_data["y_average"] else "negative"
                current_stroke_path = path_stroke_positive if current_state == "positive" else path_stroke_negative
                current_fill_path = path_fill_positive if current_state == "positive" else path_fill_negative
                if current_state != previous_state:
                    if previous_fill_path:
                        previous_fill_path.lineTo(current_x - vis_region["bin_width"], y_middle)
                        previous_stroke_path.lineTo(current_x - vis_region["bin_width"], y_middle)
                    current_fill_path.lineTo(current_x, y_middle)
                    current_stroke_path.moveTo(current_x, y_middle)
                current_fill_path.lineTo(current_x, y_bottom + bin_height)
                current_fill_path.lineTo(current_x + vis_region["bin_width"], y_bottom + bin_height)
                current_stroke_path.lineTo(current_x, y_bottom + bin_height)
                current_stroke_path.lineTo(current_x + vis_region["bin_width"], y_bottom + bin_height)
                current_x += vis_region["bin_width"]
                previous_state = current_state
                previous_fill_path = current_fill_path
                previous_stroke_path = current_stroke_path
            canvas.setStrokeColorRGB(*matplotlib.colors.hex2color(
                lovis4u.Methods.scale_lightness(self.track_data["positive_colour"], 0.8)), 1)
            canvas.drawPath(path_stroke_positive, stroke=1, fill=0)
            canvas.setStrokeColorRGB(*matplotlib.colors.hex2color(
                lovis4u.Methods.scale_lightness(self.track_data["negative_colour"], 0.8)), 1)
            canvas.drawPath(path_stroke_negative, stroke=1, fill=0)

            path_fill_positive.lineTo(vis_region["canvas_end"], y_middle)
            path_fill_negative.lineTo(vis_region["canvas_end"], y_middle)
            path_fill_positive.lineTo(vis_region["canvas_start"], y_middle)
            path_fill_negative.lineTo(vis_region["canvas_start"], y_middle)
            path_fill_positive.close()
            path_fill_negative.close()
            canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["positive_colour"]),
                                   self.prms.args["property_track_colour_alpha"])
            canvas.drawPath(path_fill_positive, stroke=0, fill=1)
            canvas.setFillColorRGB(*matplotlib.colors.hex2color(self.track_data["negative_colour"]),
                                   self.prms.args["property_track_colour_alpha"])
            canvas.drawPath(path_fill_negative, stroke=0, fill=1)

        for vis_region in self.track_data["vis_regions"]:
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
            canvas.setLineCap(0)
            canvas.setLineWidth(self.prms.args["property_track_middle_line_width"])
            canvas.setDash(1)
            canvas.line(vis_region["canvas_start"], (y_bottom + y_upper) / 2, vis_region["canvas_end"],
                        (y_bottom + y_upper) / 2)
            canvas.setDash(self.prms.args["property_track_middle_line_width"] / 2, 1)
            canvas.setLineWidth(self.prms.args["property_track_middle_line_width"] / 2)
            canvas.line(vis_region["canvas_start"], y_upper, vis_region["canvas_end"], y_upper)
            canvas.line(vis_region["canvas_start"], y_bottom, vis_region["canvas_end"], y_bottom)

        # Profile Label
        canvas.setFont(self.prms.args["property_label_font_face"], self.prms.args["property_label_font_size"])
        label_y_bottom = y_bottom + 0.3 * self.track_data["label_height"]
        canvas.setFillColorRGB(
            *matplotlib.colors.hex2color(lovis4u.Methods.scale_lightness(self.track_data["positive_colour"], 0.25)),
            1)
        # canvas.drawCentredString(self.layout["figure_x_middle_position"], label_y_bottom, self.track_data["label_text"])
        canvas.drawString(self.layout["loci_tracks_left_border"], label_y_bottom, self.track_data["label_text"])

        # Axis annotation
        canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("x_axis_line_colour", self.prms))
        canvas.setLineWidth(self.prms.args["property_track_middle_line_width"])
        canvas.setDash(1)
        canvas.setLineCap(0)
        left_border = self.layout["loci_tracks_left_border"]
        right_border = self.layout["loci_tracks_right_border"]
        axis_label_height = self.track_data["axis_labels_height"]
        canvas.setFont(self.prms.args["property_axis_font_face"], self.prms.args["property_axis_font_size"])
        full_bottom_annotation_text = f"Average: {round(self.track_data['y_average'], 2)}; Range: [" \
                                      f"{round(self.track_data['y_min'], 2)}, {round(self.track_data['y_max'], 2)}]"
        canvas.drawRightString(right_border, y_bottom + 0.3 * axis_label_height, full_bottom_annotation_text)

        return None
    except Exception as error:
        raise lovis4u.Manager.lovis4uError("Unable to draw a property track.") from error

ScaleVis

Bases: Track

ScaleVis track object that handles visualisation of scale bottom line.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class ScaleVis(Track):
    """ScaleVis track object that handles visualisation of scale bottom line.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        prms (lovis4u.Manager.Parameters): Parameters' class object.

    """

    def __init__(self, layout: dict, track_data: dict, parameters):
        """Create a LocusVis object.

        Arguments:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            track_data (dict): a dictionary with prepared track specific data.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.


        """
        super().__init__(layout, track_data, parameters)
        self.track_height = None

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Draw a ScaleVis track.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        try:
            y_upper = self.layout["current_y_coordinate"]
            y_bottom = self.layout["current_y_coordinate"] - self.track_data["track_height"]
            y_center = self.layout["current_y_coordinate"] - 0.5 * self.track_data["track_height"]

            middle_line_x_position = np.mean(self.track_data["coordinates"])

            canvas.setLineWidth(self.prms.args["scale_line_width"])
            canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("scale_line_colour", self.prms))
            canvas.setLineCap(0)
            canvas.setLineJoin(1)
            if self.track_data["style"] == "fancy":
                tick_height = self.track_data["scale_line_label_height"]

                p = canvas.beginPath()
                p.moveTo(self.track_data["coordinates"][0], y_center - 0.5 * tick_height)
                p.lineTo(self.track_data["coordinates"][0], y_center + 0.5 * tick_height)
                p.moveTo(self.track_data["coordinates"][0], y_center)
                p.lineTo(middle_line_x_position - 0.5 * self.track_data["scale_line_label_width"] -
                         self.track_data["space_width"], y_center)
                p.moveTo(self.track_data["coordinates"][1], y_center - 0.5 * tick_height)
                p.lineTo(self.track_data["coordinates"][1], y_center + 0.5 * tick_height)
                p.moveTo(self.track_data["coordinates"][1], y_center)
                p.lineTo(middle_line_x_position + 0.5 * self.track_data["scale_line_label_width"] +
                         self.track_data["space_width"], y_center)
            else:
                tick_height = self.prms.args["scale_line_tics_height"] * mm

                p = canvas.beginPath()
                p.moveTo(self.track_data["coordinates"][0], y_upper - tick_height)
                p.lineTo(self.track_data["coordinates"][0], y_upper)
                p.moveTo(self.track_data["coordinates"][0], y_upper - tick_height * 0.5)
                p.lineTo(self.track_data["coordinates"][1], y_upper - tick_height * 0.5)
                p.lineTo(self.track_data["coordinates"][1], y_upper)
                p.lineTo(self.track_data["coordinates"][1], y_upper - tick_height)
            canvas.drawPath(p, stroke=1, fill=0)
            canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("scale_line_colour", self.prms))
            canvas.setFont(self.prms.args["scale_line_label_font_face"],
                           self.track_data["scale_line_label_font_size"])
            canvas.drawCentredString(middle_line_x_position, y_bottom, self.track_data["scale_label"])
            return None
        except Exception as error:
            raise lovis4u.Manager.lovis4uError("Unable to draw a scale track.") from error

__init__(layout, track_data, parameters)

Create a LocusVis object.

Parameters:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout: dict, track_data: dict, parameters):
    """Create a LocusVis object.

    Arguments:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.


    """
    super().__init__(layout, track_data, parameters)
    self.track_height = None

draw(canvas)

Draw a ScaleVis track.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Draw a ScaleVis track.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    try:
        y_upper = self.layout["current_y_coordinate"]
        y_bottom = self.layout["current_y_coordinate"] - self.track_data["track_height"]
        y_center = self.layout["current_y_coordinate"] - 0.5 * self.track_data["track_height"]

        middle_line_x_position = np.mean(self.track_data["coordinates"])

        canvas.setLineWidth(self.prms.args["scale_line_width"])
        canvas.setStrokeColorRGB(*lovis4u.Methods.get_colour_rgba("scale_line_colour", self.prms))
        canvas.setLineCap(0)
        canvas.setLineJoin(1)
        if self.track_data["style"] == "fancy":
            tick_height = self.track_data["scale_line_label_height"]

            p = canvas.beginPath()
            p.moveTo(self.track_data["coordinates"][0], y_center - 0.5 * tick_height)
            p.lineTo(self.track_data["coordinates"][0], y_center + 0.5 * tick_height)
            p.moveTo(self.track_data["coordinates"][0], y_center)
            p.lineTo(middle_line_x_position - 0.5 * self.track_data["scale_line_label_width"] -
                     self.track_data["space_width"], y_center)
            p.moveTo(self.track_data["coordinates"][1], y_center - 0.5 * tick_height)
            p.lineTo(self.track_data["coordinates"][1], y_center + 0.5 * tick_height)
            p.moveTo(self.track_data["coordinates"][1], y_center)
            p.lineTo(middle_line_x_position + 0.5 * self.track_data["scale_line_label_width"] +
                     self.track_data["space_width"], y_center)
        else:
            tick_height = self.prms.args["scale_line_tics_height"] * mm

            p = canvas.beginPath()
            p.moveTo(self.track_data["coordinates"][0], y_upper - tick_height)
            p.lineTo(self.track_data["coordinates"][0], y_upper)
            p.moveTo(self.track_data["coordinates"][0], y_upper - tick_height * 0.5)
            p.lineTo(self.track_data["coordinates"][1], y_upper - tick_height * 0.5)
            p.lineTo(self.track_data["coordinates"][1], y_upper)
            p.lineTo(self.track_data["coordinates"][1], y_upper - tick_height)
        canvas.drawPath(p, stroke=1, fill=0)
        canvas.setFillColorRGB(*lovis4u.Methods.get_colour_rgba("scale_line_colour", self.prms))
        canvas.setFont(self.prms.args["scale_line_label_font_face"],
                       self.track_data["scale_line_label_font_size"])
        canvas.drawCentredString(middle_line_x_position, y_bottom, self.track_data["scale_label"])
        return None
    except Exception as error:
        raise lovis4u.Manager.lovis4uError("Unable to draw a scale track.") from error

Track

Parent class for visualisation Tracks.

Attributes:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • prms (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
class Track:
    """Parent class for visualisation Tracks.

    Attributes:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        prms (lovis4u.Manager.Parameters): Parameters' class object.

    """

    def __init__(self, layout: dict, track_data: dict, parameters: lovis4u.Manager.Parameters):
        """Parent's constructor for creating a Track object.

        Arguments:
            layout (dict): Layout built by CanvasManager's define_layout() method.
            track_data (dict): a dictionary with prepared track specific data.
            parameters (lovis4u.Manager.Parameters): Parameters' class object.

        """
        self.layout = layout
        self.track_data = track_data
        self.prms = parameters

    def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
        """Empy parent's method for track drawing.

        Arguments:
            canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

        Returns:
            None

        """
        pass

__init__(layout, track_data, parameters)

Parent's constructor for creating a Track object.

Parameters:

  • layout (dict) –

    Layout built by CanvasManager's define_layout() method.

  • track_data (dict) –

    a dictionary with prepared track specific data.

  • parameters (Parameters) –

    Parameters' class object.

Source code in lovis4u/Drawing.py
def __init__(self, layout: dict, track_data: dict, parameters: lovis4u.Manager.Parameters):
    """Parent's constructor for creating a Track object.

    Arguments:
        layout (dict): Layout built by CanvasManager's define_layout() method.
        track_data (dict): a dictionary with prepared track specific data.
        parameters (lovis4u.Manager.Parameters): Parameters' class object.

    """
    self.layout = layout
    self.track_data = track_data
    self.prms = parameters

draw(canvas)

Empy parent's method for track drawing.

Parameters:

  • canvas (Canvas) –

    a canvas object.

Returns:

  • None

    None

Source code in lovis4u/Drawing.py
def draw(self, canvas: reportlab.pdfgen.canvas.Canvas) -> None:
    """Empy parent's method for track drawing.

    Arguments:
        canvas (reportlab.pdfgen.canvas.Canvas): a canvas object.

    Returns:
        None

    """
    pass