\begin{figure*}[htb]
\floatconts
  {fig:resampling_simclr_framework}
  {\caption{Self-Supervised Contrastive pretraining framework. For each sample $x_{orig}$ in a batch, 4 time series are randomly sampled from it into $x_{sub}$. The resampling augmentation is then applied to create two views: $x^1$ and $x^2$. Each of the 4 time series per view per sample is individually fed to a ResNet encoder, producing 4 embeddings per view per sample into a representation space $h$ that will be used for downstream tasks. For each view, these 4 embeddings are then aggregated by averaging into a single representation ($H_1$ and $H_2$) and further projected ($Z_1$ and $Z_2$) by an MLP on which the contrastive loss is computed. The diagram illustrates the SimCLR framework but any contrastive learning framework that relies on positive pairs can be used.}}
  {\centering
    \pgfdeclarelayer{foreground}
    \pgfsetlayers{background,main,foreground}

    \begin{tikzpicture}[scale=\figscale, transform shape]

        \coordinate (plot_orig_pos) at (\plotOrigX, \plotOrigY);
        \coordinate (plot_sub_pos) at (\plotSubX, \plotSubY);
        \coordinate (plot1_pos) at (\plotOneX, \plotOneY);
        \coordinate (plot2_pos) at (\plotTwoX, \plotTwoY);

        \begin{axis}[font=\FontSizeTiny, name=axis_orig, at={(plot_orig_pos)}, anchor=center,
            width=\axisWidth, height=\axisHeight, scale only axis,
            axis lines=middle, xlabel={ }, xmin=-0.25, xmax=7, ymin=0, ymax=37,
            xtick=\empty, ytick=\empty,
            tick label style={font=\FontSizeTiny}, xlabel style={font=\FontSizeTiny}, ylabel style={font=\FontSizeTiny},
            axis line style={draw=black},
            enlarge x limits=false, axis y line*=left,
            axis background/.style={fill=white, rounded corners=3pt}
        ]
            \addplot[black, thick, mark=*, mark size=\plotMarkSize] table {\originaldata};
        \end{axis}

        \begin{scope}[on background layer]
            \node[tsBackground, opacity=0.1, fit=(axis_orig), shift={(\tsBgShiftLarge,\tsBgShiftLarge)}] (ts_orig_bg_fade_4){};
            \node[tsBackground, opacity=0.3, fit=(axis_orig), shift={(\tsBgShiftMedLarge,\tsBgShiftMedLarge)}] (ts_orig_bg_fade_3){};
            \node[tsBackground, opacity=0.5, fit=(axis_orig), shift={(\tsBgShiftMedium,\tsBgShiftMedium)}] (ts_orig_bg_fade_2){};
            \node[tsBackground, opacity=0.7, fit=(axis_orig), shift={(\tsBgShiftSmall,\tsBgShiftSmall)}] (ts_orig_bg_fade_1){};
            \node[tsBackground, opacity=0.9, fit=(axis_orig), shift={(\tsBgShiftTiny,\tsBgShiftTiny)}] (ts_orig_bg_4){};
            \node[tsBackground, fit=(axis_orig), shift={(\tsBgShiftMicro,\tsBgShiftMicro)}] (ts_orig_bg_3){};
            \node[tsBackground, fit=(axis_orig), shift={(\tsBgShiftNano,\tsBgShiftNano)}] (ts_orig_bg_2){};
            \node[tsBackground, fit=(axis_orig)] (ts_orig_bg_1){};
        \end{scope}
        \node[font=\FontSizeTiny, anchor=south, above=1pt of ts_orig_bg_fade_4.north, yshift=\axisTitleYShift] {$x_{orig}$};

        \node[draw, rectangle, rounded corners, fill=red!20, inner sep=\componentInnerSep, minimum size=\resampleNodeMinSize, font=\FontSizeSmall] (resample_node) at ($(plot_sub_pos) + (\resampleNodeOffsetX,0)$) {\rotatebox{-90}{Resampling}};
        \begin{axis}[font=\FontSizeTiny, name=axis_sub, at={(plot_sub_pos)}, anchor=center,
            width=\axisWidth, height=\axisHeight, scale only axis,
            axis lines=middle, xlabel={ }, xmin=-0.25, xmax=7, ymin=0, ymax=37,
            xtick=\empty, ytick=\empty,
            tick label style={font=\FontSizeTiny}, xlabel style={font=\FontSizeTiny}, ylabel style={font=\FontSizeTiny},
            axis line style={draw=black},
            enlarge x limits=false, axis y line*=left,
            axis background/.style={fill=white, rounded corners=3pt}
        ]
            \addplot[black, thick, mark=*, mark size=\plotMarkSize] coordinates { (0,5) (1,15) (2,32.5) (3,28.5) (4,12.5) (5,7.5) (6,11) (7,2.5) };
        \end{axis}

        \begin{axis}[font=\FontSizeTiny, name=axis1, at={(plot1_pos)}, anchor=center,
            width=\axisWidth, height=\axisHeight, scale only axis,
            axis lines=middle, xlabel={ }, xmin=-0.25, xmax=7, ymin=0, ymax=37,
            xtick=\empty, ytick=\empty,
            tick label style={font=\FontSizeTiny}, xlabel style={font=\FontSizeTiny}, ylabel style={font=\FontSizeTiny},
            axis line style={draw=black},
            enlarge x limits=false, axis y line*=left,
            axis background/.style={fill=white, rounded corners=3pt}
        ]
            \addplot[black!30!green, thick, mark=*, mark size=\plotMarkSize] coordinates { (0,5) (1,12.5) (2,28.5) (3,28.5) (4,14.4) (5,8.8) (6,5.5) (7,2.5) };
        \end{axis}

        \begin{axis}[font=\FontSizeTiny, name=axis2, at={(plot2_pos)}, anchor=center,
            width=\axisWidth, height=\axisHeight, scale only axis,
            axis lines=middle, xlabel={ }, xmin=-0.25, xmax=7, ymin=0, ymax=37,
            xtick=\empty, ytick=\empty,
            tick label style={font=\FontSizeTiny}, xlabel style={font=\FontSizeTiny}, ylabel style={font=\FontSizeTiny},
            axis line style={draw=black},
            enlarge x limits=false, axis y line*=left,
            axis background/.style={fill=white, rounded corners=3pt}
        ]
            \addplot[white!30!red, thick, mark=*, mark size=\plotMarkSize] coordinates { (0,10) (1,16) (2,30.3) (3,25) (4,15) (5,9.3) (6,9.7) (7,6.5) };
        \end{axis}

        \begin{scope}[on background layer]
            \node[
                draw, fill=blue!10, rounded corners, inner sep=\viewGenBgInnerSep, minimum height=\viewGenBgMinHeight,
                fit=(axis_sub) (resample_node) (axis1) (axis2),
                label={[anchor=south, font=\FontSizeNormal]south: Resampling View Generation}
            ] (view_generation_bg) {};
        \end{scope}

        \begin{scope}[on background layer]
            \node[tsBackground, fit=(axis_sub), shift={(\tsBgShiftTiny,\tsBgShiftTiny)}] (ts_bg_4){}; 
            \node[tsBackground, fit=(axis_sub), shift={(\tsBgShiftMicro,\tsBgShiftMicro)}] (ts_bg_3){}; 
            \node[tsBackground, fit=(axis_sub), shift={(\tsBgShiftNano,\tsBgShiftNano)}] (ts_bg_2){}; 
            \node[tsBackground, fit=(axis_sub)] (ts_bg_1){};                         
            \node[tsBackground, fit=(axis1), shift={(\tsBgShiftTiny,\tsBgShiftTiny)}] (ts1_bg_4){};
            \node[tsBackground, fit=(axis1), shift={(\tsBgShiftMicro,\tsBgShiftMicro)}] (ts1_bg_3){};
            \node[tsBackground, fit=(axis1), shift={(\tsBgShiftNano,\tsBgShiftNano)}] (ts1_bg_2){};
            \node[tsBackground, fit=(axis1)] (ts1_bg_1){};                        
            \node[tsBackground, fit=(axis2), shift={(\tsBgShiftTiny,\tsBgShiftTiny)}] (ts2_bg_4){};
            \node[tsBackground, fit=(axis2), shift={(\tsBgShiftMicro,\tsBgShiftMicro)}] (ts2_bg_3){};
            \node[tsBackground, fit=(axis2), shift={(\tsBgShiftNano,\tsBgShiftNano)}] (ts2_bg_2){};
            \node[tsBackground, fit=(axis2)] (ts2_bg_1){};                        
        \end{scope}
        \node[font=\FontSizeTiny, anchor=south, above=1pt of ts_bg_4.north, yshift=\axisTitleYShift] {$x_{sub}$};
        \node[font=\FontSizeTiny, anchor=south, above=1pt of ts1_bg_4.north, yshift=\axisTitleYShift] {$x^1 = x^1_1 .. x^1_4$};
        \node[font=\FontSizeTiny, anchor=south, above=1pt of ts2_bg_4.north, yshift=\axisTitleYShift] {$x^2 = x^2_1 .. x^2_4$};

        \path (ts_orig_bg_1.east) -- (ts_bg_1.west) node[midway, draw, rectangle, rounded corners, fill=red!20, font=\FontSizeSmall, align=center, inner sep=\componentInnerSep] (sample_node) {Sample\\4 timeseries};
        \draw[->, thick] (ts_orig_bg_1.east) -- (sample_node.west);
        \draw[->, thick] (sample_node.east) -- (ts_bg_1.west);

        \tikzstyle{projector} = [draw, rectangle, rounded corners, minimum width=\projectorMinWidth, minimum height=\projectorMinHeight, fill=orange!20, align=center]
        \tikzstyle{loss} = [draw, circle, minimum size=\resampleNodeMinSize, fill=red!20, align=center]
        \tikzstyle{aggregator} = [draw, rectangle, rounded corners, fill=red!20, minimum width=\aggregatorMinWidth, inner sep=\componentInnerSep]
        \tikzstyle{componentstyle} = [draw, rectangle, rounded corners, fill=orange!10, inner sep=\componentStyleInnerSep]

        \begin{scope}[shift={($(view_generation_bg.east)+(\simclrScopeShiftX,0)$)}] 

            \pic (enc1) at (0,\encOneY) {encoderPic}; 
            \node[font=\FontSizeSmall, align=center, text width=\encLabelTextWidth, below=\encLabelOffsetY of enc1-center] {ResNet\\Encoder};
            \node[font=\FontSizeSmall, right=\hSetLabelOffsetX of enc1-right, align=center] (h1_set_label) {$h^1_1 .. h^1_4$};            
            \draw[->, thick] (enc1-right) -- (h1_set_label.west);           
            \node[font=\FontSizeSmall, aggregator, right=\aggOffsetX of h1_set_label] (agg1) {\rotatebox{-90}{Aggregate}};          
            \draw[->, thick] (h1_set_label.east) -- (agg1.west);        
            \node[font=\FontSizeSmall, right=\HLabelOffsetX of agg1] (H1_label_node) {$H_1$};
            \draw[->, thick] (agg1.east) -- (H1_label_node.west);
            \node[font=\FontSizeSmall, projector, right=\projOffsetX of H1_label_node] (proj1) {\rotatebox{-90}{MLP Proj.}};
            \node[font=\FontSizeSmall, right=\ZLabelOffsetX of proj1] (Z1_label) {$Z_1$};
            \coordinate (Z1_fit_extend) at ($(Z1_label.east) + (\ZLabelFitExtend,0)$); 
            \draw[->, thick] (H1_label_node.east) -- (proj1.west);

            \pic (enc2) at (0,\encTwoY) {encoderPic};
            \node[font=\FontSizeSmall, align=center, text width=\encLabelTextWidth, below=\encLabelOffsetY of enc2-center] {ResNet\\Encoder};
            \node[font=\FontSizeSmall, right=\hSetLabelOffsetX of enc2-right, align=center] (h2_set_label) {$h^2_1 .. h^2_4$};            
            \draw[->, thick] (enc2-right) -- (h2_set_label.west);
            \node[font=\FontSizeSmall, aggregator, right=\aggOffsetX of h2_set_label] (agg2) {\rotatebox{-90}{Aggregate}};          
            \draw[->, thick] (h2_set_label.east) -- (agg2.west);
            \node[font=\FontSizeSmall, right=\HLabelOffsetX of agg2] (H2_label_node) {$H_2$};
            \draw[->, thick] (agg2.east) -- (H2_label_node.west);
            \node[font=\FontSizeSmall, projector, right=\projOffsetX of H2_label_node] (proj2) {\rotatebox{-90}{MLP Proj.}};
            \node[font=\FontSizeSmall, right=\ZLabelOffsetX of proj2] (Z2_label) {$Z_2$};
            \coordinate (Z2_fit_extend) at ($(Z2_label.east) + (\ZLabelFitExtend,0)$); 
            \draw[->, thick] (H2_label_node.east) -- (proj2.west);

            \draw[->, thick] ($(ts_bg_1.east) + (\arrowToResampleNodeOffsetX,0)$) -- (resample_node.west);
            \draw[->, thick, rounded corners=3pt] (resample_node) |- (ts1_bg_1);
            \draw[->, thick, rounded corners=3pt] (resample_node) |- (ts2_bg_1);
            \draw[->, thick] ($(ts1_bg_1.east) + (\arrowToEncoderOffsetX,0)$) -- (enc1-left);
            \draw[->, thick] ($(ts2_bg_1.east) + (\arrowToEncoderOffsetX,0)$) -- (enc2-left);

            \node[font=\FontSizeSmall, draw, rectangle, rounded corners, fill=red!20, align=center, inner sep=\componentInnerSep] (contrastive_loss) at ($(Z1_label.east)!0.5!(Z2_label.east) + (\contrastiveLossOffsetX,0)$) {Contrastive\\Loss};
            \draw[->, thick, rounded corners=3pt] (Z1_label.east) -| (contrastive_loss);
            \draw[->, thick, rounded corners=3pt] (Z2_label.east) -| (contrastive_loss);
            \draw[->, thick] (proj2.east) -- (Z2_label.west);
            \draw[->, thick] (proj1.east) -- (Z1_label.west);

            \begin{scope}[on background layer]
                \node[draw, fill=my_green!10, rounded corners, inner sep=\simclrBgInnerSep, minimum height=\simclrBgMinHeight,
                      fit=(enc1-left)(h1_set_label)(agg1)(H1_label_node)(proj1)(Z1_label)(Z1_fit_extend)
                          (enc2-left)(h2_set_label)(agg2)(H2_label_node)(proj2)(Z2_label)(Z2_fit_extend)
                          (contrastive_loss.east),
                      label={[anchor=south, font=\FontSizeNormal]south:SimCLR Framework}
                ] (simclr_bg_scoped) {};
            \end{scope}
        \end{scope}

        \node (legend_nontrainable) [font=\FontSizeTiny, rectangle, draw, fill=red!20, rounded corners, align=center, inner sep=\componentInnerSep, anchor=north west, at={($(axis_orig.south west) + (0,\legendNontrainableOffsetY)$)}] {Non-trainable\\components};
        \node (legend_trainable) [font=\FontSizeTiny, rectangle, draw, fill=orange!20, rounded corners, align=center, inner sep=\componentInnerSep, right=\legendTrainableSpacing of legend_nontrainable, anchor=west] {Trainable\\components};
    \end{tikzpicture}
  }
\end{figure*}
