From bdcbd52c6236cd0156293b72dc0d8a5b064f4a14 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Fri, 17 Sep 2021 15:41:52 +0200 Subject: [PATCH 1/7] Add Image trace style --- .../CommonAbstractions/StyleParams.fs | 38 ++++++++++ src/Plotly.NET/Traces/Trace2D.fs | 71 +++++++++++++++++++ 2 files changed, 109 insertions(+) diff --git a/src/Plotly.NET/CommonAbstractions/StyleParams.fs b/src/Plotly.NET/CommonAbstractions/StyleParams.fs index f0e5dc362..85ff7f72c 100644 --- a/src/Plotly.NET/CommonAbstractions/StyleParams.fs +++ b/src/Plotly.NET/CommonAbstractions/StyleParams.fs @@ -314,6 +314,44 @@ module StyleParam = // #C# //-------------------------- + + [] + type ColorModel = + | RGB + | RGBA + | RGBA256 + | HSL + | HSLA + + static member toString = function + | RGB -> "rgb" + | RGBA -> "rgba" + | RGBA256 -> "rgba256" + | HSL -> "hsl" + | HSLA -> "hsla" + + static member convert = ColorModel.toString >> box + override this.ToString() = this |> ColorModel.toString + member this.Convert() = this |> ColorModel.convert + + [] + type ColorComponentBound = + | RGB of R:int * G:int * B:int + | RGBA of R:int * G:int * B:int * A:int + | RGBA256 of int*int*int*int + | HSL of H:int * S:int * L:int + | HSLA of H:int * S:int * L:int * A:int + + static member convert = function + | RGB (r,g,b) -> [r;g;b] |> box + | RGBA (r,g,b,a) -> [r;g;b;a] |> box + | RGBA256 (i,d,k,m) -> [i;d;k;m] |> box + | HSL (h,s,l) -> [h;s;l] |> box + | HSLA (h,s,l,a) -> [h;s;l;a] |> box + + member this.Convert() = this |> ColorComponentBound.convert + + [] type ClickToShow = | False | OnOff | OnOut diff --git a/src/Plotly.NET/Traces/Trace2D.fs b/src/Plotly.NET/Traces/Trace2D.fs index eeb4bffaf..760c8d276 100644 --- a/src/Plotly.NET/Traces/Trace2D.fs +++ b/src/Plotly.NET/Traces/Trace2D.fs @@ -946,6 +946,77 @@ type Trace2DStyle() = // out -> heatmap ) + + // Applies the styles of heatmap to TraceObjects + static member Image + ( + [] ?Name : string, + [] ?Visible : StyleParam.Visible, + [] ?ShowLegend : bool, + [] ?LegendRank : int, + [] ?LegendGroup : string, + [] ?LegendGroupTitle : Title, + [] ?Opacity : float, + [] ?Ids : seq<#IConvertible>, + [] ?X : seq<#IConvertible>, + [] ?X0 : #IConvertible, + [] ?DX : #IConvertible, + [] ?Y : seq<#IConvertible>, + [] ?Y0 : #IConvertible, + [] ?DY : #IConvertible, + [] ?Z : seq<#seq<#IConvertible>>, + [] ?Source : string, + [] ?Text : seq<#IConvertible>, + [] ?HoverText : string, + [] ?HoverInfo : StyleParam.HoverInfo, + [] ?HoverTemplate : string, + [] ?Meta : string, + [] ?CustomData : seq<#IConvertible>, + [] ?XAxis : StyleParam.LinearAxisId, + [] ?YAxis : StyleParam.LinearAxisId, + [] ?ColorModel : StyleParam.ColorModel, + [] ?ZMax : StyleParam.ColorComponentBound, + [] ?ZMin : StyleParam.ColorComponentBound, + [] ?ZSmooth : StyleParam.SmoothAlg, + [] ?HoverLabel : Hoverlabel, + [] ?UIRevision : string + ) = + (fun (image: ('T :> Trace)) -> + + Name |> DynObj.setValueOpt image "name" + Visible |> DynObj.setValueOptBy image "visible" StyleParam.Visible.convert + ShowLegend |> DynObj.setValueOpt image "showlegend" + LegendRank |> DynObj.setValueOpt image "legendrank" + LegendGroup |> DynObj.setValueOpt image "legendgroup" + LegendGroupTitle |> DynObj.setValueOpt image "legendgrouptitle" + Opacity |> DynObj.setValueOpt image "opacity" + Ids |> DynObj.setValueOpt image "ids" + X |> DynObj.setValueOpt image "x" + X0 |> DynObj.setValueOpt image "x0" + DX |> DynObj.setValueOpt image "dx" + Y |> DynObj.setValueOpt image "y" + Y0 |> DynObj.setValueOpt image "y0" + DY |> DynObj.setValueOpt image "dy" + Z |> DynObj.setValueOpt image "z" + Source |> DynObj.setValueOpt image "source" + Text |> DynObj.setValueOpt image "text" + HoverText |> DynObj.setValueOpt image "hovertext" + HoverInfo |> DynObj.setValueOptBy image "hoverinfo" StyleParam.HoverInfo.convert + HoverTemplate |> DynObj.setValueOpt image "hovertemplate" + Meta |> DynObj.setValueOpt image "meta" + CustomData |> DynObj.setValueOpt image "customdata" + XAxis |> DynObj.setValueOptBy image "xaxis" StyleParam.LinearAxisId.convert + YAxis |> DynObj.setValueOptBy image "yaxis" StyleParam.LinearAxisId.convert + ColorModel |> DynObj.setValueOptBy image "colormodel" StyleParam.ColorModel.convert + ZMax |> DynObj.setValueOptBy image "zmax" StyleParam.ColorComponentBound.convert + ZMin |> DynObj.setValueOptBy image "zmin" StyleParam.ColorComponentBound.convert + ZSmooth |> DynObj.setValueOptBy image "zsmooth" StyleParam.SmoothAlg.convert + HoverLabel |> DynObj.setValueOpt image "hoverlabel" + UIRevision |> DynObj.setValueOpt image "uirevision" + + // out -> + image + ) static member Contour From 1a35c0264a1c22a9b010ae7baa9c649b35fae6f3 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Sun, 19 Sep 2021 09:15:40 +0200 Subject: [PATCH 2/7] Add Chart.Image --- src/Plotly.NET/ChartAPI/Chart2D.fs | 72 ++++++++++++++++++++++++++++++ src/Plotly.NET/Playground.fsx | 32 ++++++++++++- src/Plotly.NET/Traces/Trace2D.fs | 2 +- 3 files changed, 104 insertions(+), 2 deletions(-) diff --git a/src/Plotly.NET/ChartAPI/Chart2D.fs b/src/Plotly.NET/ChartAPI/Chart2D.fs index 09b8382ec..48d9a5dce 100644 --- a/src/Plotly.NET/ChartAPI/Chart2D.fs +++ b/src/Plotly.NET/ChartAPI/Chart2D.fs @@ -1436,6 +1436,78 @@ module Chart2D = Chart.renderHeatmapTrace useWebGL style + [] + static member Image + ( + [] ?Z : seq<#seq<#seq>>, + [] ?Source : string, + [] ?Name : string, + [] ?ShowLegend : bool, + [] ?Opacity : float, + [] ?Ids : seq<#IConvertible>, + [] ?X : seq<#IConvertible>, + [] ?Y : seq<#IConvertible>, + [] ?ColorModel : StyleParam.ColorModel, + [] ?ZMax : StyleParam.ColorComponentBound, + [] ?ZMin : StyleParam.ColorComponentBound, + [] ?ZSmooth : StyleParam.SmoothAlg + ) = + + Trace2D.initImage ( + Trace2DStyle.Image ( + ?Z = Z , + ?Source = Source , + ?Name = Name , + ?ShowLegend = ShowLegend, + ?Opacity = Opacity , + ?Ids = Ids , + ?X = X , + ?Y = Y , + ?ColorModel = ColorModel, + ?ZMax = ZMax , + ?ZMin = ZMin , + ?ZSmooth = ZSmooth + ) + ) + |> GenericChart.ofTraceObject + + [] + static member Image + ( + z : seq<#seq>, + [] ?Name : string, + [] ?ShowLegend : bool, + [] ?Opacity : float, + [] ?Ids : seq<#IConvertible>, + [] ?X : seq<#IConvertible>, + [] ?Y : seq<#IConvertible>, + [] ?ZMax : StyleParam.ColorComponentBound, + [] ?ZMin : StyleParam.ColorComponentBound, + [] ?ZSmooth : StyleParam.SmoothAlg + ) = + + let z' = + z + |> Seq.map (Seq.map (fun argb -> seq{int argb.R; int argb.G; int argb.B; int argb.A;})) + + + Trace2D.initImage ( + Trace2DStyle.Image ( + Z = z' , + ?Name = Name , + ?ShowLegend = ShowLegend, + ?Opacity = Opacity , + ?Ids = Ids , + ?X = X , + ?Y = Y , + ColorModel = StyleParam.ColorModel.RGBA, + ?ZMax = ZMax , + ?ZMin = ZMin , + ?ZSmooth = ZSmooth + ) + ) + |> GenericChart.ofTraceObject + /// Shows a graphical representation of data where the individual values contained in a matrix are represented as colors. [] static member Contour(data:seq<#seq<#IConvertible>>, diff --git a/src/Plotly.NET/Playground.fsx b/src/Plotly.NET/Playground.fsx index 6f6148345..3f18ed085 100644 --- a/src/Plotly.NET/Playground.fsx +++ b/src/Plotly.NET/Playground.fsx @@ -27,10 +27,10 @@ #load "Margin.fs" #load "Domain.fs" #load "Shape.fs" +#load "Hoverlabel.fs" #load "Annotation.fs" #load "LayoutGrid.fs" #load "Legend.fs" -#load "Hoverlabel.fs" #load "TickFormatStop.fs" #load "ColorBar.fs" #load "Rangebreak.fs" @@ -153,6 +153,36 @@ open FSharpAux open System open System.IO +let imagebase64 = + System.Convert.ToBase64String(File.ReadAllBytes(@"C:\Users\schne\Pictures\Untitled.jpg")) + +Chart.Image( + Source=($"data:image/jpg;base64,{imagebase64}") +) +|> Chart.show + +let colors = [ + [[0 ;0 ;255]; [255;255;0 ]; [0 ;0 ;255]] + [[255;0 ;0 ]; [255;0 ;255]; [255;0 ;255]] + [[0 ;255;0 ]; [0 ;255;255]; [255;0 ;0 ]] +] + +Chart.Image( + Z=colors +) +|> Chart.show + +let argbs = [ + [(0 ,0 ,255); (0 ,255,255); (0 ,0 ,255)] |> List.map (fun (r,g,b) -> ARGB.fromRGB r g b ) + [(255,0 ,0 ); (255,0 ,255); (0 ,0 ,0 )] |> List.map (fun (r,g,b) -> ARGB.fromRGB r g b ) + [(0 ,255,0 ); (255,255,0 ); (0 ,0 ,255)] |> List.map (fun (r,g,b) -> ARGB.fromRGB r g b ) +] + +Chart.Image( + argbs +) +|> Chart.show + Chart.ScatterTernary( A = [1; 2; 3], B = [3; 2; 1], diff --git a/src/Plotly.NET/Traces/Trace2D.fs b/src/Plotly.NET/Traces/Trace2D.fs index 760c8d276..cc2fba620 100644 --- a/src/Plotly.NET/Traces/Trace2D.fs +++ b/src/Plotly.NET/Traces/Trace2D.fs @@ -964,7 +964,7 @@ type Trace2DStyle() = [] ?Y : seq<#IConvertible>, [] ?Y0 : #IConvertible, [] ?DY : #IConvertible, - [] ?Z : seq<#seq<#IConvertible>>, + [] ?Z : #seq<#seq<#seq>>, [] ?Source : string, [] ?Text : seq<#IConvertible>, [] ?HoverText : string, From c85460040e09df053bc5d697d35282453a6a4782 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Sun, 19 Sep 2021 09:53:56 +0200 Subject: [PATCH 3/7] Add LayoutImage object --- .../CommonAbstractions/StyleParams.fs | 16 ++++ src/Plotly.NET/Layout/Layout.fs | 10 ++- .../ObjectAbstractions/Common/LayoutImage.fs | 87 +++++++++++++++++++ src/Plotly.NET/Plotly.NET.fsproj | 1 + 4 files changed, 111 insertions(+), 3 deletions(-) create mode 100644 src/Plotly.NET/Layout/ObjectAbstractions/Common/LayoutImage.fs diff --git a/src/Plotly.NET/CommonAbstractions/StyleParams.fs b/src/Plotly.NET/CommonAbstractions/StyleParams.fs index 85ff7f72c..6fe57ac23 100644 --- a/src/Plotly.NET/CommonAbstractions/StyleParams.fs +++ b/src/Plotly.NET/CommonAbstractions/StyleParams.fs @@ -1119,7 +1119,23 @@ module StyleParam = // #L# //-------------------------- + + /// Specifies whether shapes are drawn below or above traces. Default is Above + [] + type LayoutImageSizing = + | Fill + | Contain + | Stretch + static member toString = function + | Fill -> "fill" + | Contain -> "contain" + | Stretch -> "stretch" + + static member convert = LayoutImageSizing.toString >> box + override this.ToString() = this |> LayoutImageSizing.toString + member this.Convert() = this |> LayoutImageSizing.convert + /// Specifies whether shapes are drawn below or above traces. Default is Above [] type Layer = diff --git a/src/Plotly.NET/Layout/Layout.fs b/src/Plotly.NET/Layout/Layout.fs index 1f93891b7..050d940ae 100644 --- a/src/Plotly.NET/Layout/Layout.fs +++ b/src/Plotly.NET/Layout/Layout.fs @@ -76,7 +76,8 @@ type Layout() = [] ?ExtendIcicleColors : bool, [] ?IcicleColorWay : Color, [] ?Annotations : seq, - [] ?Shapes : seq + [] ?Shapes : seq, + [] ?Images : seq ) = Layout() |> Layout.style @@ -145,7 +146,8 @@ type Layout() = ?ExtendIcicleColors = ExtendIcicleColors , ?IcicleColorWay = IcicleColorWay , ?Annotations = Annotations , - ?Shapes = Shapes + ?Shapes = Shapes , + ?Images = Images ) // Applies the styles to Layout() @@ -215,7 +217,8 @@ type Layout() = [] ?ExtendIcicleColors : bool, [] ?IcicleColorWay : Color, [] ?Annotations : seq, - [] ?Shapes : seq + [] ?Shapes : seq, + [] ?Images : seq ) = (fun (layout:Layout) -> @@ -284,6 +287,7 @@ type Layout() = IcicleColorWay |> DynObj.setValueOpt layout "iciclecolorway" Annotations |> DynObj.setValueOpt layout "annotations" Shapes |> DynObj.setValueOpt layout "shapes" + Images |> DynObj.setValueOpt layout "images" layout ) diff --git a/src/Plotly.NET/Layout/ObjectAbstractions/Common/LayoutImage.fs b/src/Plotly.NET/Layout/ObjectAbstractions/Common/LayoutImage.fs new file mode 100644 index 000000000..e14ac7740 --- /dev/null +++ b/src/Plotly.NET/Layout/ObjectAbstractions/Common/LayoutImage.fs @@ -0,0 +1,87 @@ +namespace Plotly.NET.LayoutObjects + +open Plotly.NET +open DynamicObj +open System +open System.Runtime.InteropServices + +/// Dimensions type inherits from dynamic object +type LayoutImage () = + inherit DynamicObj () + + static member init + ( + ?Layer : StyleParam.Layer, + ?Name : string, + ?Opacity : float, + ?SizeX : int, + ?SizeY : int, + ?Sizing : StyleParam.LayoutImageSizing, + ?Source : string, + ?TemplateItemname : string, + ?Visible : bool, + ?X : #IConvertible, + ?XAnchor : StyleParam.XAnchorPosition, + ?XRef : string, + ?Y : #IConvertible, + ?YAnchor : StyleParam.YAnchorPosition, + ?YRef : string + ) = + LayoutImage () + |> LayoutImage.style + ( + ?Layer = Layer , + ?Name = Name , + ?Opacity = Opacity , + ?SizeX = SizeX , + ?SizeY = SizeY , + ?Sizing = Sizing , + ?Source = Source , + ?TemplateItemname = TemplateItemname, + ?Visible = Visible , + ?X = X , + ?XAnchor = XAnchor , + ?XRef = XRef , + ?Y = Y , + ?YAnchor = YAnchor , + ?YRef = YRef + ) + + static member style + ( + ?Layer : StyleParam.Layer, + ?Name : string, + ?Opacity : float, + ?SizeX : int, + ?SizeY : int, + ?Sizing : StyleParam.LayoutImageSizing, + ?Source : string, + ?TemplateItemname : string, + ?Visible : bool, + ?X : #IConvertible, + ?XAnchor : StyleParam.XAnchorPosition, + ?XRef : string, + ?Y : #IConvertible, + ?YAnchor : StyleParam.YAnchorPosition, + ?YRef : string + ) = + (fun (layoutImage:LayoutImage) -> + + Layer |> DynObj.setValueOptBy layoutImage "layer" StyleParam.Layer.convert + Name |> DynObj.setValueOpt layoutImage "name" + Opacity |> DynObj.setValueOpt layoutImage "opacity" + SizeX |> DynObj.setValueOpt layoutImage "sizex" + SizeY |> DynObj.setValueOpt layoutImage "sizey" + Sizing |> DynObj.setValueOptBy layoutImage "sizing" StyleParam.LayoutImageSizing.convert + Source |> DynObj.setValueOpt layoutImage "source" + TemplateItemname |> DynObj.setValueOpt layoutImage "templateitemname" + Visible |> DynObj.setValueOpt layoutImage "visible" + X |> DynObj.setValueOpt layoutImage "x" + XAnchor |> DynObj.setValueOptBy layoutImage "xanchor" StyleParam.XAnchorPosition.convert + XRef |> DynObj.setValueOpt layoutImage "xref" + Y |> DynObj.setValueOpt layoutImage "y" + YAnchor |> DynObj.setValueOptBy layoutImage "yanchor" StyleParam.YAnchorPosition.convert + YRef |> DynObj.setValueOpt layoutImage "yref" + + layoutImage + ) \ No newline at end of file diff --git a/src/Plotly.NET/Plotly.NET.fsproj b/src/Plotly.NET/Plotly.NET.fsproj index 55b0fe870..5ca1a0253 100644 --- a/src/Plotly.NET/Plotly.NET.fsproj +++ b/src/Plotly.NET/Plotly.NET.fsproj @@ -41,6 +41,7 @@ + From acb3b9c9bde707d75119c44650876884467262eb Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Sun, 19 Sep 2021 09:54:22 +0200 Subject: [PATCH 4/7] Add Chart.withLayoutImage(s), adapt Chart.withShape(s) --- .../CSharpLayer/GenericChartExtensions.fs | 35 ++++--- src/Plotly.NET/ChartAPI/Chart.fs | 98 ++++++++++++++++--- src/Plotly.NET/Playground.fsx | 18 ++++ 3 files changed, 123 insertions(+), 28 deletions(-) diff --git a/src/Plotly.NET/CSharpLayer/GenericChartExtensions.fs b/src/Plotly.NET/CSharpLayer/GenericChartExtensions.fs index ea2cf11c0..bae45b65c 100644 --- a/src/Plotly.NET/CSharpLayer/GenericChartExtensions.fs +++ b/src/Plotly.NET/CSharpLayer/GenericChartExtensions.fs @@ -493,12 +493,15 @@ module GenericChartExtensions = member this.WithConfig (config:Config) = GenericChart.setConfig config this + [] + [] + member this.WithAnnotation(annotation:Annotation, []?Append:bool) = + this |> Chart.withAnnotation(annotation, ?Append = Append) + [] [] - member this.WithAnnotations(annotations:seq) = - this - |> GenericChart.mapLayout - (Layout.style (Annotations = annotations)) + member this.WithAnnotations(annotations:Annotation seq, []?Append:bool) = + this |> Chart.withAnnotations(annotations, ?Append = Append) // Set the title of a Chart [] @@ -574,19 +577,13 @@ module GenericChartExtensions = //(`x0`,`y0`), (`x1`,`y0`), (`x1`,`y1`), (`x0`,`y1`), (`x0`,`y0`) [] [] - member this.WithShape(shape:Shape) = - let layout = - GenericChart.getLayout this - |> Layout.style (Shapes=[shape]) - GenericChart.setLayout layout this + member this.WithShape(shape:Shape, []?Append:bool) = + this |> Chart.withShape(shape, ?Append = Append) [] [] - member this.WithShapes(shapes:Shape seq) = - let layout = - GenericChart.getLayout this - |> Layout.style (Shapes=shapes) - GenericChart.setLayout layout this + member this.WithShapes(shapes:Shape seq, []?Append:bool) = + this |> Chart.withShapes(shapes, ?Append = Append) // ############################################################ // ####################### Apply to DisplayOptions @@ -664,3 +661,13 @@ module GenericChartExtensions = member this.WithTernary(ternary:Ternary, [] ?Id) = this |> Chart.withTernary(ternary,?Id=Id) + + [] + [] + member this.WithLayoutImage(image:LayoutImage, []?Append:bool) = + this |> Chart.withLayoutImage(image, ?Append = Append) + + [] + [] + member this.WithLayoutImages(images:seq, []?Append:bool) = + this |> Chart.withLayoutImages(images, ?Append = Append) \ No newline at end of file diff --git a/src/Plotly.NET/ChartAPI/Chart.fs b/src/Plotly.NET/ChartAPI/Chart.fs index d9c82bc3b..adef70048 100644 --- a/src/Plotly.NET/ChartAPI/Chart.fs +++ b/src/Plotly.NET/ChartAPI/Chart.fs @@ -823,6 +823,14 @@ type Chart = |> GenericChart.mapLayout (Layout.style (Annotations = annotations')) + [] + static member withAnnotation + ( + annotation: Annotation, + [] ?Append: bool + ) = + Chart.withAnnotations([annotation], ?Append = Append) + // Set the title of a Chart [] static member withTitle(title,[] ?TitleFont) = @@ -911,23 +919,45 @@ type Chart = //Specifies the shape type to be drawn. If "line", a line is drawn from (`x0`,`y0`) to (`x1`,`y1`) If "circle", a circle is drawn from //((`x0`+`x1`)/2, (`y0`+`y1`)/2)) with radius (|(`x0`+`x1`)/2 - `x0`|, |(`y0`+`y1`)/2 -`y0`)|) If "rect", a rectangle is drawn linking //(`x0`,`y0`), (`x1`,`y0`), (`x1`,`y1`), (`x0`,`y1`), (`x0`,`y0`) - [] - static member withShape(shape:Shape) = - (fun (ch:GenericChart) -> - let layout = - GenericChart.getLayout ch - |> Layout.style (Shapes=[shape]) - GenericChart.setLayout layout ch) + + /// + /// + /// + /// The shapes to add to the input charts layout + /// If true, the input annotations will be appended to existing annotations, otherwise existing annotations will be removed (default: true) + [] + static member withShapes + ( + shapes:seq, + [] ?Append: bool + ) = + let append = defaultArg Append true + fun (ch:GenericChart) -> + + let shapes' = - [] - static member withShapes(shapes:Shape seq) = - (fun (ch:GenericChart) -> - let layout = - GenericChart.getLayout ch - |> Layout.style (Shapes=shapes) - GenericChart.setLayout layout ch) + if append then + + let layout = GenericChart.getLayout ch + + layout.TryGetTypedValue>("shapes") + |> Option.defaultValue Seq.empty + |> Seq.append shapes + + else shapes + + ch + |> GenericChart.mapLayout + (Layout.style (Shapes = shapes')) + [] + static member withShape + ( + shape: Shape, + [] ?Append: bool + ) = + Chart.withShapes([shape], ?Append = Append) // ####################### /// Create a combined chart with the given charts merged @@ -1491,3 +1521,43 @@ type Chart = GenericChart.setLayout updatedLayout ch ) + + /// + /// + /// + /// The images to add to the input charts layout + /// If true, the input images will be appended to existing annotations, otherwise existing annotations will be removed (default: true) + [] + static member withLayoutImages + ( + images:seq, + [] ?Append: bool + ) = + let append = defaultArg Append true + + fun (ch:GenericChart) -> + + let images' = + + if append then + + let layout = GenericChart.getLayout ch + + layout.TryGetTypedValue>("images") + |> Option.defaultValue Seq.empty + |> Seq.append images + + else images + + ch + |> GenericChart.mapLayout + (Layout.style (Images = images')) + + [] + static member withLayoutImage + ( + image: LayoutImage, + [] ?Append: bool + ) = + + Chart.withLayoutImages([image], ?Append = Append) \ No newline at end of file diff --git a/src/Plotly.NET/Playground.fsx b/src/Plotly.NET/Playground.fsx index 3f18ed085..b579e132a 100644 --- a/src/Plotly.NET/Playground.fsx +++ b/src/Plotly.NET/Playground.fsx @@ -16,6 +16,7 @@ #I "Layout/ObjectAbstractions/Common" +#load "LayoutImage.fs" #load "Button.fs" #load "RangeSelector.fs" #load "RangeSlider.fs" @@ -153,6 +154,23 @@ open FSharpAux open System open System.IO +Chart.Line([0.; 0.5; 1.; 2.; 2.2], y=[1.23; 2.5; 0.42; 3.; 1.]) +|> Chart.withLayoutImage( + LayoutImage.init( + Source="https://fsharp.org/img/logo/fsharp.svg", + XRef="x", + YRef="y", + X=0, + Y=3, + SizeX=2, + SizeY=2, + Sizing=StyleParam.LayoutImageSizing.Stretch, + Opacity=0.5, + Layer=StyleParam.Layer.Below + ) +) +|> Chart.show + let imagebase64 = System.Convert.ToBase64String(File.ReadAllBytes(@"C:\Users\schne\Pictures\Untitled.jpg")) From b29a51bcfd89470e0b7a14b7335b00ca51778225 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Sun, 19 Sep 2021 10:08:43 +0200 Subject: [PATCH 5/7] Add layoutimage docs --- Plotly.NET.sln | 1 + docs/01_5_layout_images.fsx | 76 ++++++++++++++++++++++++ docs/10_0_ternary_line_scatter_plots.fsx | 2 +- docs/10_1_styling_ternary_layouts.fsx | 4 +- 4 files changed, 80 insertions(+), 3 deletions(-) create mode 100644 docs/01_5_layout_images.fsx diff --git a/Plotly.NET.sln b/Plotly.NET.sln index ba7fa9ca6..d68e54db6 100644 --- a/Plotly.NET.sln +++ b/Plotly.NET.sln @@ -49,6 +49,7 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "docs", "docs", "{7B09CC0A-F docs\01_2_multiple-charts.fsx = docs\01_2_multiple-charts.fsx docs\01_3_shapes.fsx = docs\01_3_shapes.fsx docs\01_4_annotations.fsx = docs\01_4_annotations.fsx + docs\01_5_layout_images.fsx = docs\01_5_layout_images.fsx docs\02_0_line-scatter-plots.fsx = docs\02_0_line-scatter-plots.fsx docs\02_1_bar-and-column-charts.fsx = docs\02_1_bar-and-column-charts.fsx docs\02_2_area-plots.fsx = docs\02_2_area-plots.fsx diff --git a/docs/01_5_layout_images.fsx b/docs/01_5_layout_images.fsx new file mode 100644 index 000000000..72b680721 --- /dev/null +++ b/docs/01_5_layout_images.fsx @@ -0,0 +1,76 @@ +(** +--- +title: Layout images +category: Chart Layout +categoryindex: 2 +index: 6 +--- +*) + +(*** hide ***) + +(*** condition: prepare ***) +#r "nuget: Newtonsoft.JSON, 12.0.3" +#r "nuget: DynamicObj" +#r "../bin/Plotly.NET/netstandard2.0/Plotly.NET.dll" + +(*** condition: ipynb ***) +#if IPYNB +#r "nuget: Plotly.NET, {{fsdocs-package-version}}" +#r "nuget: Plotly.NET.Interactive, {{fsdocs-package-version}}" +#endif // IPYNB + +(** +# Annotations + +[![Binder]({{root}}img/badge-binder.svg)](https://mybinder.org/v2/gh/plotly/Plotly.NET/gh-pages?filepath={{fsdocs-source-basename}}.ipynb)  +[![Script]({{root}}img/badge-script.svg)]({{root}}{{fsdocs-source-basename}}.fsx)  +[![Notebook]({{root}}img/badge-notebook.svg)]({{root}}{{fsdocs-source-basename}}.ipynb) + + +*Summary:* This example shows how to create Images and add them to the Charts in F#. + +let's first create some data for the purpose of creating example charts: + +*) + +open Plotly.NET + +let x = [1.; 2.; 3.; 4.; 5.; 6.; 7.; 8.; 9.; 10.; ] +let y' = [2.; 1.5; 5.; 1.5; 3.; 2.5; 2.5; 1.5; 3.5; 1.] + +(** +use the `LayoutImage.init` function to generate an image, and either the `Chart.withLayoutImage` or the `Chart.withLayoutImages` function to add +multiple annotations at once. + +*) + +open Plotly.NET.LayoutObjects + +let image = + LayoutImage.init( + Source="https://fsharp.org/img/logo/fsharp.svg", + XRef="x", + YRef="y", + X=0, + Y=3, + SizeX=2, + SizeY=2, + Sizing=StyleParam.LayoutImageSizing.Stretch, + Opacity=0.5, + Layer=StyleParam.Layer.Below + ) + +let imageChart = + Chart.Line(x,y',Name="line") + |> Chart.withLayoutImage(image) + +(*** condition: ipynb ***) +#if IPYNB +imageChart +#endif // IPYNB + +(***hide***) +imageChart |> GenericChart.toChartHTML +(***include-it-raw***) + diff --git a/docs/10_0_ternary_line_scatter_plots.fsx b/docs/10_0_ternary_line_scatter_plots.fsx index 3f9af2161..668ca0c73 100644 --- a/docs/10_0_ternary_line_scatter_plots.fsx +++ b/docs/10_0_ternary_line_scatter_plots.fsx @@ -2,7 +2,7 @@ --- title: Ternary line and scatter plots category: Ternary Plots -categoryindex: 10 +categoryindex: 11 index: 1 --- *) diff --git a/docs/10_1_styling_ternary_layouts.fsx b/docs/10_1_styling_ternary_layouts.fsx index 2e4cfca38..3d55169aa 100644 --- a/docs/10_1_styling_ternary_layouts.fsx +++ b/docs/10_1_styling_ternary_layouts.fsx @@ -2,8 +2,8 @@ --- title: Styling ternary layouts category: Ternary Plots -categoryindex: 9 -index: 3 +categoryindex: 11 +index: 2 --- *) From e69aaa5f00e13c5e9d3468b0a6646210fae8b913 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Mon, 20 Sep 2021 10:45:26 +0200 Subject: [PATCH 6/7] Add Image docs --- Plotly.NET.sln | 1 + docs/02_8_Images.fsx | 130 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 131 insertions(+) create mode 100644 docs/02_8_Images.fsx diff --git a/Plotly.NET.sln b/Plotly.NET.sln index d68e54db6..da836e59c 100644 --- a/Plotly.NET.sln +++ b/Plotly.NET.sln @@ -58,6 +58,7 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "docs", "docs", "{7B09CC0A-F docs\02_5_pie-doughnut-charts.fsx = docs\02_5_pie-doughnut-charts.fsx docs\02_6_table.fsx = docs\02_6_table.fsx docs\02_7_heatmaps.fsx = docs\02_7_heatmaps.fsx + docs\02_8_Images.fsx = docs\02_8_Images.fsx docs\03_0_3d-scatter-plots.fsx = docs\03_0_3d-scatter-plots.fsx docs\03_1_3d-surface-plots.fsx = docs\03_1_3d-surface-plots.fsx docs\03_2_3d-mesh-plots.fsx = docs\03_2_3d-mesh-plots.fsx diff --git a/docs/02_8_Images.fsx b/docs/02_8_Images.fsx new file mode 100644 index 000000000..e00f11128 --- /dev/null +++ b/docs/02_8_Images.fsx @@ -0,0 +1,130 @@ +(** +--- +title: Images +category: Simple Charts +categoryindex: 3 +index: 9 +--- +*) + +(*** hide ***) + +(*** condition: prepare ***) +#r "nuget: Newtonsoft.JSON, 12.0.3" +#r "nuget: DynamicObj" +#r "../bin/Plotly.NET/netstandard2.0/Plotly.NET.dll" + +(*** condition: ipynb ***) +#if IPYNB +#r "nuget: Plotly.NET, {{fsdocs-package-version}}" +#r "nuget: Plotly.NET.Interactive, {{fsdocs-package-version}}" +#endif // IPYNB + +(** +# Images + +[![Binder]({{root}}img/badge-binder.svg)](https://mybinder.org/v2/gh/plotly/Plotly.NET/gh-pages?filepath={{fsdocs-source-basename}}.ipynb)  +[![Script]({{root}}img/badge-script.svg)]({{root}}{{fsdocs-source-basename}}.fsx)  +[![Notebook]({{root}}img/badge-notebook.svg)]({{root}}{{fsdocs-source-basename}}.ipynb) + +*Summary:* This example shows how to create image charts in F#. + +There are multiple ways of generating image charts: + - From 3 Dimensional color collections, where the inner arrays contain 3 (color dimensions without alpha channel) or 4 (color dimensions and alpha channel) values. The color model can be set separately as shown below. + - From a 2 dimensional collection Plotly.NETs `ARGB` type that represents rgba values + - From a base64 encoded image data source + +## Creating Image charts from raw color arrays +*) + +// 3d collection containing color values +open Plotly.NET + +let colors = [ + [[0 ;0 ;255]; [255;255;0 ]; [0 ;0 ;255]] + [[255;0 ;0 ]; [255;0 ;255]; [255;0 ;255]] + [[0 ;255;0 ]; [0 ;255;255]; [255;0 ;0 ]] +] + +let imageRaw = + Chart.Image(Z=colors) + |> Chart.withTitle "Image chart from raw color component arrays" + +(*** condition: ipynb ***) +#if IPYNB +imageRaw +#endif // IPYNB + +(***hide***) +imageRaw |> GenericChart.toChartHTML +(***include-it-raw***) + +(** +To change the color model to HSL for example, add the `ColorModel` argument: +*) + +let imageRawHSL = + Chart.Image(Z=colors, ColorModel=StyleParam.ColorModel.HSL) + |> Chart.withTitle "HSL color model" + +(*** condition: ipynb ***) +#if IPYNB +imageRawHSL +#endif // IPYNB + +(***hide***) +imageRawHSL |> GenericChart.toChartHTML +(***include-it-raw***) + +(** +## Creating Image charts from ARGB arrays + +Note that this way of creating image charts uses the RGBA color model. +*) + +let argbs = [ + [ColorKeyword.AliceBlue ; ColorKeyword.CornSilk ; ColorKeyword.LavenderBlush ] |> List.map ARGB.fromKeyword + [ColorKeyword.DarkGray ; ColorKeyword.Snow ; ColorKeyword.MidnightBlue ] |> List.map ARGB.fromKeyword + [ColorKeyword.LightSteelBlue; ColorKeyword.DarkKhaki; ColorKeyword.LightAkyBlue ] |> List.map ARGB.fromKeyword +] + +let imageARGB = + Chart.Image(argbs) + |> Chart.withTitle "ARGB image chart" + +(*** condition: ipynb ***) +#if IPYNB +imageARGB +#endif // IPYNB + +(***hide***) +imageARGB |> GenericChart.toChartHTML +(***include-it-raw***) + +(** +## Creating Image charts from base64 encoded images +*) +open System +open System.IO + +let imageSource = $@"{__SOURCE_DIRECTORY__}/img/logo.png" + +let base64String = + imageSource + |> File.ReadAllBytes + |> System.Convert.ToBase64String + +let logoImage = + Chart.Image( + Source=($"data:image/jpg;base64,{base64String}") + ) + |> Chart.withTitle "This is Plotly.NET:" + +(*** condition: ipynb ***) +#if IPYNB +logoImage +#endif // IPYNB + +(***hide***) +logoImage |> GenericChart.toChartHTML +(***include-it-raw***) From e17e533e7c4abf75da6c901502e838e30ea777f6 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Mon, 20 Sep 2021 11:27:32 +0200 Subject: [PATCH 7/7] Add image chart tests --- .../HtmlCodegen/SimpleCharts.fs | 87 +++++++++++++++++++ 1 file changed, 87 insertions(+) diff --git a/tests/Plotly.NET.Tests/HtmlCodegen/SimpleCharts.fs b/tests/Plotly.NET.Tests/HtmlCodegen/SimpleCharts.fs index cce9b92e8..5ea7634dd 100644 --- a/tests/Plotly.NET.Tests/HtmlCodegen/SimpleCharts.fs +++ b/tests/Plotly.NET.Tests/HtmlCodegen/SimpleCharts.fs @@ -514,4 +514,91 @@ let ``Heatmap charts`` = "var layout = {\"width\":700,\"height\":500,\"margin\":{\"l\":200.0}};" |> chartGeneratedContains heatmapStyledChart ); + ] + + +let colors = [ + [[0 ;0 ;255]; [255;255;0 ]; [0 ;0 ;255]] + [[255;0 ;0 ]; [255;0 ;255]; [255;0 ;255]] + [[0 ;255;0 ]; [0 ;255;255]; [255;0 ;0 ]] +] + +let imageRawChart = + Chart.Image(Z=colors) + |> Chart.withTitle "Image chart from raw color component arrays" + +let imageRawHSLChart = + Chart.Image(Z=colors, ColorModel=StyleParam.ColorModel.HSL) + |> Chart.withTitle "HSL color model" + +let argbs = [ + [ColorKeyword.AliceBlue ; ColorKeyword.CornSilk ; ColorKeyword.LavenderBlush ] |> List.map ARGB.fromKeyword + [ColorKeyword.DarkGray ; ColorKeyword.Snow ; ColorKeyword.MidnightBlue ] |> List.map ARGB.fromKeyword + [ColorKeyword.LightSteelBlue; ColorKeyword.DarkKhaki; ColorKeyword.LightAkyBlue ] |> List.map ARGB.fromKeyword +] + +let imageARGBChart = + Chart.Image(argbs) + |> Chart.withTitle "ARGB image chart" + +open System.IO + +let imageSource = $@"{__SOURCE_DIRECTORY__}../../../../docs/img/logo.png" + +let base64String = + imageSource + |> File.ReadAllBytes + |> System.Convert.ToBase64String + +let logoImageChart = + Chart.Image( + Source=($"data:image/jpg;base64,{base64String}") + ) + |> Chart.withTitle "This is Plotly.NET:" + +imageRawChart |> Chart.show +imageRawHSLChart |> Chart.show +imageARGBChart |> Chart.show +logoImageChart |> Chart.show + +[] +let ``Image charts`` = + testList "SimpleCharts.Image charts" [ + testCase "Image raw data" ( fun () -> + """var data = [{"type":"image","z":[[[0,0,255],[255,255,0],[0,0,255]],[[255,0,0],[255,0,255],[255,0,255]],[[0,255,0],[0,255,255],[255,0,0]]]}];""" + |> chartGeneratedContains imageRawChart + ); + testCase "Image raw layout" ( fun () -> + """var layout = {"title":{"text":"Image chart from raw color component arrays"}};""" + |> chartGeneratedContains imageRawChart + ); + + testCase "Image raw hsl data" ( fun () -> + """var data = [{"type":"image","z":[[[0,0,255],[255,255,0],[0,0,255]],[[255,0,0],[255,0,255],[255,0,255]],[[0,255,0],[0,255,255],[255,0,0]]],"colormodel":"hsl"}];""" + |> chartGeneratedContains imageRawHSLChart + ); + testCase "Image raw hsl layout" ( fun () -> + """var layout = {"title":{"text":"HSL color model"}};""" + |> chartGeneratedContains imageRawHSLChart + ); + + testCase "Image ARGB data" ( fun () -> + """var data = [{"type":"image","z":[[[240,248,255,255],[255,248,220,255],[255,240,245,255]],[[169,169,169,255],[255,250,250,255],[25,25,112,255]],[[176,196,222,255],[189,183,107,255],[135,206,250,255]]],"colormodel":"rgba"}];""" + |> chartGeneratedContains imageARGBChart + ); + testCase "Image ARGB layout" ( fun () -> + """var layout = {"title":{"text":"ARGB image chart"}};""" + |> chartGeneratedContains imageARGBChart + ); + + testCase "Image base64 data" ( fun () -> + """var data = [{"type":"image","source":""}];""" + |> chartGeneratedContains logoImageChart + ); + testCase "Image base64 layout" ( fun () -> + """var layout = {"title":{"text":"This is Plotly.NET:"}};""" + |> chartGeneratedContains logoImageChart + ); + + ] \ No newline at end of file