diff --git a/.gitignore b/.gitignore index b31a349ad..e878845f5 100644 --- a/.gitignore +++ b/.gitignore @@ -187,3 +187,7 @@ temp/gh-pages /.fsdocs /tmp/watch /output +/src/Plotly.NET.ImageExport/.local-chromium +/src/Plotly.NET.ImageExport/testrenders +/src/Plotly.NET.Interactive/.local-chromium/ +/docs/.local-chromium/ diff --git a/Plotly.NET.sln b/Plotly.NET.sln index da7661dcc..9568ab057 100644 --- a/Plotly.NET.sln +++ b/Plotly.NET.sln @@ -51,6 +51,7 @@ EndProject Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "content", "content", "{60FB82C0-F472-494E-BCF7-7B3C54212406}" ProjectSection(SolutionItems) = preProject docs\0_0_basics.fsx = docs\0_0_basics.fsx + docs\0_1_image-export.fsx = docs\0_1_image-export.fsx docs\1_0_axis-styling.fsx = docs\1_0_axis-styling.fsx docs\1_1_errorbars.fsx = docs\1_1_errorbars.fsx docs\1_2_multiple-charts.fsx = docs\1_2_multiple-charts.fsx @@ -102,6 +103,8 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "img", "img", "{CDB973F2-0F6 docs\img\logo.png = docs\img\logo.png EndProjectSection EndProject +Project("{6EC3EE1D-3C4E-46DD-8F32-0CC8E7565705}") = "Plotly.NET.ImageExport", "src\Plotly.NET.ImageExport\Plotly.NET.ImageExport.fsproj", "{6CFC629E-1A0C-4EF3-8495-BA00A356A381}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -127,6 +130,12 @@ Global {0F135E3B-B0E1-42A9-B180-18C0221DC7B8}.Dotnet|Any CPU.Build.0 = Debug|Any CPU {0F135E3B-B0E1-42A9-B180-18C0221DC7B8}.Release|Any CPU.ActiveCfg = Release|Any CPU {0F135E3B-B0E1-42A9-B180-18C0221DC7B8}.Release|Any CPU.Build.0 = Release|Any CPU + {6CFC629E-1A0C-4EF3-8495-BA00A356A381}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {6CFC629E-1A0C-4EF3-8495-BA00A356A381}.Debug|Any CPU.Build.0 = Debug|Any CPU + {6CFC629E-1A0C-4EF3-8495-BA00A356A381}.Dotnet|Any CPU.ActiveCfg = Debug|Any CPU + {6CFC629E-1A0C-4EF3-8495-BA00A356A381}.Dotnet|Any CPU.Build.0 = Debug|Any CPU + {6CFC629E-1A0C-4EF3-8495-BA00A356A381}.Release|Any CPU.ActiveCfg = Release|Any CPU + {6CFC629E-1A0C-4EF3-8495-BA00A356A381}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE diff --git a/build.fsx.lock b/build.fsx.lock index cfe87901e..73316be79 100644 --- a/build.fsx.lock +++ b/build.fsx.lock @@ -144,11 +144,11 @@ NUGET FSharp.Core (>= 4.7.2) FParsec (1.1.1) FSharp.Core (>= 4.3.4) - FSharp.Compiler.Service (39.0) - FSharp.Core (5.0.1) - Microsoft.Build.Framework (>= 16.6) - Microsoft.Build.Tasks.Core (>= 16.6) - Microsoft.Build.Utilities.Core (>= 16.6) + FSharp.Compiler.Service (40.0) + FSharp.Core (5.0.2) + Microsoft.Build.Framework (>= 16.9) + Microsoft.Build.Tasks.Core (>= 16.9) + Microsoft.Build.Utilities.Core (>= 16.9) System.Buffers (>= 4.5.1) System.Collections.Immutable (>= 5.0) System.Diagnostics.Process (>= 4.3) @@ -173,7 +173,7 @@ NUGET FSharp.Control.Reactive (5.0.2) FSharp.Core (>= 4.7.2) System.Reactive (>= 5.0) - FSharp.Core (5.0.1) + FSharp.Core (5.0.2) Microsoft.Build (16.10) Microsoft.Build.Framework (16.10) System.Security.Permissions (>= 4.7) @@ -213,27 +213,27 @@ NUGET System.Security.AccessControl (>= 5.0) System.Security.Principal.Windows (>= 5.0) Mono.Posix.NETStandard (1.0) - MSBuild.StructuredLogger (2.1.500) + MSBuild.StructuredLogger (2.1.507) Microsoft.Build (>= 16.4) Microsoft.Build.Framework (>= 16.4) Microsoft.Build.Tasks.Core (>= 16.4) Microsoft.Build.Utilities.Core (>= 16.4) Newtonsoft.Json (13.0.1) - NuGet.Common (5.9.1) - NuGet.Frameworks (>= 5.9.1) - NuGet.Configuration (5.9.1) - NuGet.Common (>= 5.9.1) + NuGet.Common (5.10) + NuGet.Frameworks (>= 5.10) + NuGet.Configuration (5.10) + NuGet.Common (>= 5.10) System.Security.Cryptography.ProtectedData (>= 4.4) - NuGet.Frameworks (5.9.1) - NuGet.Packaging (5.9.1) + NuGet.Frameworks (5.10) + NuGet.Packaging (5.10) Newtonsoft.Json (>= 9.0.1) - NuGet.Configuration (>= 5.9.1) - NuGet.Versioning (>= 5.9.1) + NuGet.Configuration (>= 5.10) + NuGet.Versioning (>= 5.10) System.Security.Cryptography.Cng (>= 5.0) System.Security.Cryptography.Pkcs (>= 5.0) - NuGet.Protocol (5.9.1) - NuGet.Packaging (>= 5.9.1) - NuGet.Versioning (5.9.1) + NuGet.Protocol (5.10) + NuGet.Packaging (>= 5.10) + NuGet.Versioning (5.10) Octokit (0.50) runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl (4.3.3) runtime.debian.9-x64.runtime.native.System.Security.Cryptography.OpenSsl (4.3.3) diff --git a/docs/0_1_image-export.fsx b/docs/0_1_image-export.fsx new file mode 100644 index 000000000..6df06f805 --- /dev/null +++ b/docs/0_1_image-export.fsx @@ -0,0 +1,185 @@ +(** +--- +title: Static image export +category: General +categoryindex: 1 +index: 2 +--- +*) + + +(*** hide ***) + +(*** condition: prepare ***) +#r "nuget: Newtonsoft.JSON, 12.0.3" +#r "nuget: PuppeteerSharp" +#r "../bin/Plotly.NET/netstandard2.0/Plotly.NET.dll" +#r "../bin/Plotly.NET.ImageExport/netstandard2.0/Plotly.NET.ImageExport.dll" + +(*** condition: ipynb ***) +#if IPYNB +#r "nuget: Plotly.NET, {{fsdocs-package-version}}" +#r "nuget: Plotly.NET.Interactive, {{fsdocs-package-version}}" +#r "nuget: Plotly.NET.ImageExport, {{fsdocs-package-version}}" +#endif // IPYNB + + +(** +[![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) + +# Static image export + +### Table of contents + +- [Saving static images](#Saving-static-images) +- [Generating URIs for static chart images](#Generating-URIs-for-static-chart-images) +- [Including static images in dotnet interactive notebooks](#Including-static-images-in-dotnet-interactive-notebooks) + +As Plotly.NET generates static html pages that contain charts rendered by plotly.js, static image export needs a lot more overhead under the hood +than you might expect. The underlying renderer needs to execute javascript, leading to the usage of headless browsers. + +The package `Plotly.NET.ImageExport` contains extensions for Plotly.NET to render static images. It is designed with extensibility in mind and +it is very easy to add a new rendering engine. The current engines are provided: + +| Rendering engine | Type | Prerequisites | +|-|-|-| +| [PuppeteerSharp](https://github.com/hardkoded/puppeteer-sharp) | headless browser | [read more here](https://github.com/hardkoded/puppeteer-sharp#prerequisites) | + +## Saving static images + +By referencing the `Plotly.NET.ImageExport` package, you get access to: + + - jpg via `Chart.SaveJPG` + - png via `Chart.SavePNG` + - svg via `Chart.SaveSVG` + +(and Extensions for C# style fluent interfaces by opening the `GenericChartExtensions` namespace) + +The parameters for all three functions are exactly the same. +*) + +open Plotly.NET +open Plotly.NET.ImageExport + +let exampleChart = + Chart.Histogram2dContour( + [1.;2.;2.;4.;5.], + [1.;2.;2.;4.;5.] + ) + +(***do-not-eval***) +exampleChart +|> Chart.saveJPG( + "/your/path/without/extension/here", + Width=300, + Height=300 +) + +(*** condition: ipynb ***) +#if IPYNB +let imgString = $"""""" +DisplayExtensions.DisplayAs(imgString,"text/html") +#endif // IPYNB + +(***hide***) +$"""""" +(***include-it-raw***) + +(** +## Generating URIs for static chart images + +By referencing the `Plotly.NET.ImageExport` package, you get access to: + + - jpg via `Chart.toBase64JPGString` + - png via `Chart.toBase64PNGString` + - svg via `Chart.toSVGString` + +(and Extensions for C# style fluent interfaces by opening the `GenericChartExtensions` namespace) + +*) + +let base64JPG = + exampleChart + |> Chart.toBase64JPGString( + Width=300, + Height=300 + ) + +(** +It is very easy to construct a html tag that includes this image via a base64 uri. For SVGs, +not even that is necessary and just the SVG string can be used. +*) + +(***do-not-eval***) +$"""""" + +(*** condition: ipynb ***) +#if IPYNB +let imgString = $"""""" +DisplayExtensions.DisplayAs(imgString,"text/html") +#endif // IPYNB + +(***hide***) +$"""""" + +(***include-it-raw***) + +(** +SVGs can be included without the image tag: +*) + +let svgString = + exampleChart + |> Chart.toSVGString( + Width=300, + Height=300 + ) + +svgString.Substring(0,300) +|> printfn "%s" + +(***include-output***) + +(** +In fact, the images shown on this site are included just the same way. + +## Including static images in dotnet interactive notebooks + +To include the images in dotnet interactive, convert them to html tags as above and include them via +dotnet interactive's `DisplayAs` function. The content type for PNG/JPB is "text/html", and "image/svg+xml" for SVG. +*) + +let base64PNG = + exampleChart + |> Chart.toBase64PNGString( + Width=300, + Height=300 + ) + +let svgString2 = + exampleChart + |> Chart.toSVGString( + Width=300, + Height=300 + ) + +// DisplayExtensions.DisplayAs(base64PNG,"text/html") +// DisplayExtensions.DisplayAs(svgString,"image/svg+xml") + +(*** condition: ipynb ***) +#if IPYNB +DisplayExtensions.DisplayAs(base64PNG,"text/html") +DisplayExtensions.DisplayAs(svgString,"image/svg+xml") +#endif // IPYNB \ No newline at end of file diff --git a/src/Plotly.NET.ImageExport/ChartExtensions.fs b/src/Plotly.NET.ImageExport/ChartExtensions.fs new file mode 100644 index 000000000..b4c7dffbc --- /dev/null +++ b/src/Plotly.NET.ImageExport/ChartExtensions.fs @@ -0,0 +1,251 @@ +namespace Plotly.NET.ImageExport + +open System +open System.IO + +open Plotly.NET +open GenericChart +open System.Runtime.InteropServices +open System.Runtime.CompilerServices + +/// Extensions methods from Plotly.NET.ImageExport for the Chart module, supporting the fluent pipeline style +[] +module ChartExtensions = + + type internal RenderOptions + ( + ?EngineType: ExportEngine, + ?Width : int, + ?Height : int + ) = + member _.Engine = (defaultArg EngineType ExportEngine.PuppeteerSharp) |> ExportEngine.getEngine + member _.Width = defaultArg Width 600 + member _.Height = defaultArg Height 600 + + type Chart with + + /// + /// Returns an async function that converts a GenericChart to a base64 encoded JPG string + /// + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member toBase64JPGStringAsync + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + let opts = RenderOptions(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + fun (gChart:GenericChart) -> opts.Engine.RenderJPGAsync(opts.Width, opts.Height, gChart) + + /// + /// Returns a function that converts a GenericChart to a base64 encoded JPG string + /// + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member toBase64JPGString + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + fun (gChart:GenericChart) -> + gChart + |> Chart.toBase64JPGStringAsync(?EngineType=EngineType, ?Width=Width, ?Height=Height) + |> Async.RunSynchronously + + /// + /// Returns an async function that saves a GenericChart as JPG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member saveJPGAsync + ( + path: string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + let opts = RenderOptions(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + fun (gChart:GenericChart) -> opts.Engine.SaveJPGAsync(path, opts.Width, opts.Height, gChart) + + /// + /// Returns a function that saves a GenericChart as JPG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member saveJPG + ( + path: string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + fun (gChart:GenericChart) -> + gChart + |> Chart.saveJPGAsync(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + |> Async.RunSynchronously + + /// + /// Returns an async function that converts a GenericChart to a base64 encoded PNG string + /// + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member toBase64PNGStringAsync + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + let opts = RenderOptions(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + fun (gChart:GenericChart) -> opts.Engine.RenderPNGAsync(opts.Width, opts.Height, gChart) + + /// + /// Returns a function that converts a GenericChart to a base64 encoded PNG string + /// + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member toBase64PNGString + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + fun (gChart:GenericChart) -> + gChart + |> Chart.toBase64PNGStringAsync(?EngineType=EngineType, ?Width=Width, ?Height=Height) + |> Async.RunSynchronously + + /// + /// Returns an async function that saves a GenericChart as PNG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member savePNGAsync + ( + path: string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + let opts = RenderOptions(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + fun (gChart:GenericChart) -> opts.Engine.SavePNGAsync(path, opts.Width, opts.Height, gChart) + + + /// + /// Returns a function that saves a GenericChart as PNG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member savePNG + ( + path: string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + fun (gChart:GenericChart) -> + gChart + |> Chart.savePNGAsync(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + |> Async.RunSynchronously + + /// + /// Returns an async function that converts a GenericChart to a SVG string + /// + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member toSVGStringAsync + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + let opts = RenderOptions(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + fun (gChart:GenericChart) -> opts.Engine.RenderSVGAsync(opts.Width, opts.Height, gChart) + + /// + /// Returns a function that converts a GenericChart to a SVG string + /// + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member toSVGString + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + fun (gChart:GenericChart) -> + gChart + |> Chart.toSVGStringAsync(?EngineType=EngineType, ?Width=Width, ?Height=Height) + |> Async.RunSynchronously + + /// + /// Returns an async function that saves a GenericChart as SVG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member saveSVGAsync + ( + path: string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + let opts = RenderOptions(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + fun (gChart:GenericChart) -> opts.Engine.SaveSVGAsync(path, opts.Width, opts.Height, gChart) + + /// + /// Returns a function that saves a GenericChart as SVG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + static member saveSVG + ( + path: string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + fun (gChart:GenericChart) -> + gChart + |> Chart.saveSVGAsync(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + |> Async.RunSynchronously \ No newline at end of file diff --git a/src/Plotly.NET.ImageExport/ExportEngine.fs b/src/Plotly.NET.ImageExport/ExportEngine.fs new file mode 100644 index 000000000..3a0a2da8c --- /dev/null +++ b/src/Plotly.NET.ImageExport/ExportEngine.fs @@ -0,0 +1,12 @@ +namespace Plotly.NET.ImageExport + +///DU containing the available static image export engines for Plotly.NET +type ExportEngine = + /// Using this engine will use PuppeteerSharp with a Chromium headless browser to render GenericCharts from Plotly.NET. + | PuppeteerSharp + + static member getEngine (engineType: ExportEngine) = + match engineType with + ///This engine uses PuppeteerSharp to render GenericCharts as static images using a chromium headless browser. + | PuppeteerSharp -> + PuppeteerSharpRenderer() :> IGenericChartRenderer \ No newline at end of file diff --git a/src/Plotly.NET.ImageExport/GenericChartExtensions.fs b/src/Plotly.NET.ImageExport/GenericChartExtensions.fs new file mode 100644 index 000000000..604fc6f10 --- /dev/null +++ b/src/Plotly.NET.ImageExport/GenericChartExtensions.fs @@ -0,0 +1,224 @@ +namespace Plotly.NET.ImageExport + +open Plotly.NET +open GenericChart +open System +open System.Runtime.InteropServices +open System.Runtime.CompilerServices + +///Extension methods for providing a Plotly.NET.ImageExport fluent interface pattern for C# +[] +module GenericChartExtensions = + + type GenericChart with + + /// + /// Converts the GenericChart to a base64 encoded JPG string (async) + /// + /// + /// + /// + [] + [] + member this.ToBase64JPGStringAsync + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + this |> Chart.toBase64JPGStringAsync(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Converts the GenericChart to a base64 encoded JPG string + /// + /// + /// + /// + [] + [] + member this.ToBase64JPGString + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + this |> Chart.toBase64JPGString(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Saves the GenericChart as JPG image (async) + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + [] + member this.SaveJPGAsync + ( + path:string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + this |> Chart.saveJPGAsync(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Saves the GenericChart as JPG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + [] + member this.SaveJPG + ( + path:string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + this |> Chart.saveJPG(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Converts the GenericChart to a base64 encoded PNG string (async) + /// + /// + /// + /// + [] + [] + member this.ToBase64PNGStringAsync + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + this |> Chart.toBase64PNGStringAsync(?EngineType = EngineType, ?Width=Width, ?Height=Height) + + /// + /// Converts the GenericChart to a base64 encoded PNG string + /// + /// + /// + /// + [] + [] + member this.ToBase64PNGString + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + this |> Chart.toBase64PNGString(?EngineType = EngineType, ?Width=Width, ?Height=Height) + + /// + /// Saves the GenericChart as PNG image (async) + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + [] + member this.SavePNGAsync + ( + path:string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + this |> Chart.savePNG(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Saves the GenericChart as PNG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + [] + member this.SavePNG + ( + path:string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + this |> Chart.savePNG(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Converts the GenericChart to a SVG string (async) + /// + /// + /// + /// + [] + [] + member this.ToSVGStringAsync + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + this |> Chart.toSVGStringAsync(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Converts the GenericChart to a SVG string + /// + /// + /// + /// + [] + [] + member this.ToSVGString + ( + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + + this |> Chart.toSVGString(?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Saves the GenericChart as SVG image (async) + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + [] + member this.SaveSVGAsync + ( + path:string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + this |> Chart.saveSVGAsync(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + + /// + /// Saves the GenericChart as SVG image + /// + /// The path (without extension) where the image will be saved + /// The Render engine to use + /// width of the resulting image + /// height of the resulting image + [] + [] + member this.SaveSVG + ( + path:string, + [] ?EngineType: ExportEngine, + [] ?Width : int, + [] ?Height : int + ) = + this |> Chart.saveSVG(path, ?EngineType=EngineType, ?Width=Width, ?Height=Height) + \ No newline at end of file diff --git a/src/Plotly.NET.ImageExport/IGenericChartRenderer.fs b/src/Plotly.NET.ImageExport/IGenericChartRenderer.fs new file mode 100644 index 000000000..f366d9c15 --- /dev/null +++ b/src/Plotly.NET.ImageExport/IGenericChartRenderer.fs @@ -0,0 +1,36 @@ +namespace Plotly.NET.ImageExport +open Plotly.NET +/// +/// Interface for Engines that render Plotly.NET's GenericChart to static images. +/// +type IGenericChartRenderer = + + ///Async function that returns a base64 encoded string representing the input chart as JPG file with the given width and height + abstract member RenderJPGAsync: int * int * GenericChart.GenericChart -> Async + ///Function that returns a base64 encoded string representing the input chart as JPG file with the given width and height + abstract member RenderJPG: int * int * GenericChart.GenericChart -> string + + ///Async function that saves the input chart as JPG file with the given width and height at the given path + abstract member SaveJPGAsync: string * int * int * GenericChart.GenericChart -> Async + ///Function that saves the input chart as JPG file with the given width and height at the given path + abstract member SaveJPG: string * int * int * GenericChart.GenericChart -> unit + + ///Async function that returns a base64 encoded string representing the input chart as PNG file with the given width and height + abstract member RenderPNGAsync: int * int * GenericChart.GenericChart -> Async + ///Function that returns a base64 encoded string representing the input chart as PNG file with the given width and height + abstract member RenderPNG: int * int * GenericChart.GenericChart -> string + + ///Async function that saves the input chart as PNG file with the given width and height at the given path + abstract member SavePNGAsync: string * int * int * GenericChart.GenericChart -> Async + ///Function that saves the input chart as PNG file with the given width and height at the given path + abstract member SavePNG: string * int * int * GenericChart.GenericChart -> unit + + ///Async function that returns a string representing the input chart as SVG file with the given width and height + abstract member RenderSVGAsync: int * int * GenericChart.GenericChart -> Async + ///Function that returns string representing the input chart as SVG file with the given width and height + abstract member RenderSVG: int * int * GenericChart.GenericChart -> string + + ///Async function that saves the input chart as SVG file with the given width and height at the given path + abstract member SaveSVGAsync: string * int * int * GenericChart.GenericChart -> Async + ///Function that saves the input chart as SVG file with the given width and height at the given path + abstract member SaveSVG: string * int * int * GenericChart.GenericChart -> unit diff --git a/src/Plotly.NET.ImageExport/Playground.fsx b/src/Plotly.NET.ImageExport/Playground.fsx new file mode 100644 index 000000000..8f0975d50 --- /dev/null +++ b/src/Plotly.NET.ImageExport/Playground.fsx @@ -0,0 +1,165 @@ +#r "nuget: FSharp.Data" +#r "nuget: Deedle" +#r "nuget: FSharpAux" +#r "nuget: Newtonsoft.Json, 12.0.3" + +#load "../Plotly.NET/StyleParams.fs" +#load "../Plotly.NET/DynamicObj.fs" +#load "../Plotly.NET/Frame.fs" +#load "../Plotly.NET/Colors.fs" +#load "../Plotly.NET/StockData.fs" +#load "../Plotly.NET/Font.fs" +#load "../Plotly.NET/Pathbar.fs" +#load "../Plotly.NET/TreemapTiling.fs" +#load "../Plotly.NET/Colorbar.fs" +#load "../Plotly.NET/RangeSlider.fs" +#load "../Plotly.NET/Light.fs" +#load "../Plotly.NET/Legend.fs" +#load "../Plotly.NET/Contours.fs" +#load "../Plotly.NET/Dimensions.fs" +#load "../Plotly.NET/Domain.fs" +#load "../Plotly.NET/Line.fs" +#load "../Plotly.NET/WaterfallConnector.fs" +#load "../Plotly.NET/FunnelConnector.fs" +#load "../Plotly.NET/Box.fs" +#load "../Plotly.NET/Meanline.fs" +#load "../Plotly.NET/Marker.fs" +#load "../Plotly.NET/Hoverlabel.fs" +#load "../Plotly.NET/Axis.fs" +#load "../Plotly.NET/Bins.fs" +#load "../Plotly.NET/Cumulative.fs" +#load "../Plotly.NET/Scene.fs" +#load "../Plotly.NET/Selected.fs" +#load "../Plotly.NET/Shape.fs" +#load "../Plotly.NET/Error.fs" +#load "../Plotly.NET/Table.fs" +#load "../Plotly.NET/Trace.fs" +#load "../Plotly.NET/Trace3d.fs" +#load "../Plotly.NET/GeoProjection.fs" +#load "../Plotly.NET/Geo.fs" +#load "../Plotly.NET/MapboxLayerSymbol.fs" +#load "../Plotly.NET/MapboxLayer.fs" +#load "../Plotly.NET/Mapbox.fs" +#load "../Plotly.NET/LayoutGrid.fs" +#load "../Plotly.NET/Annotation.fs" +#load "../Plotly.NET/Layout.fs" +#load "../Plotly.NET/Template.fs" +#load "../Plotly.NET/Config.fs" +#load "../Plotly.NET/DisplayOptions.fs" +#load "../Plotly.NET/GenericChart.fs" +#load "../Plotly.NET/Chart.fs" +#load "../Plotly.NET/ChartExtensions.fs" +#load "../Plotly.NET/GenericChartExtensions.fs" +#load "../Plotly.NET/CandelstickExtension.fs" +#load "../Plotly.NET/SankeyExtension.fs" + +open Plotly.NET +open GenericChart + +open FSharp.Data +open Newtonsoft.Json +open System.Text +open System.IO +open Deedle +open FSharpAux + + +#r "nuget: PuppeteerSharp" + +open PuppeteerSharp + +#load "IGenericChartRenderer.fs" +#load "PuppeteerSharpRenderer.fs" +#load "ExportEngine.fs" +#load "ChartExtensions.fs" +#load "GenericChartExtensions.fs" + +open Plotly.NET.ImageExport +open GenericChartExtensions + +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.] +let xyz = Seq.zip3 x y x + +let simpleChart = Chart.Point([1.,1.]) + +let complexChart = + Chart.Grid( + [ + [Chart.Line(x,y); Chart.Line(x,y); Chart.Line(x,y)] + [Chart.Histogram2dContour(x,y); Chart.Point(x,y); Chart.Point(x,y)] + [Chart.Spline(x,y); Chart.Spline(x,y); Chart.Spline(x,y)] + ] + ) + +complexChart +|> Chart.Show + +simpleChart +|> Chart.saveJPG ( + __SOURCE_DIRECTORY__ + "/testrenders/simple", + EngineType = ExportEngine.PuppeteerSharp, + Width= 1000, + Height= 1000 +) + +open FSharp.Data +open Deedle + +let dataDensityMapbox = + Http.RequestString "https://raw.githubusercontent.com/plotly/datasets/master/earthquakes-23k.csv" + |> fun d -> Frame.ReadCsvString(d,true,separators=",") + +dataDensityMapbox.Print() + +let lonDensity = dataDensityMapbox.["Longitude"] |> Series.values +let latDensity = dataDensityMapbox.["Latitude"] |> Series.values +let magnitudes = dataDensityMapbox.["Magnitude"] |> Series.values + +let map = + Chart.DensityMapbox( + lonDensity, + latDensity, + Z = magnitudes, + Radius=8., + Colorscale=StyleParam.Colorscale.Viridis + ) + |> Chart.withMapbox( + Mapbox.init( + Style = StyleParam.MapboxStyle.StamenTerrain, + Center = (60.,30.) + ) + ) + + +map.SaveSVG(__SOURCE_DIRECTORY__ + "/testrenders/map") + +complexChart.SaveJPG (__SOURCE_DIRECTORY__ + "/testrenders/complex") + +simpleChart +|> Chart.savePNG (__SOURCE_DIRECTORY__ + "/testrenders/simple") + +complexChart + .WithTitle("soos") + .SavePNG(__SOURCE_DIRECTORY__ + "/testrenders/complex") + +simpleChart +|> Chart.saveSVG (__SOURCE_DIRECTORY__ + "/testrenders/simple") + +complexChart.SaveSVG (__SOURCE_DIRECTORY__ + "/testrenders/complex") + +let jpgString = + Chart.Point([1.,1.]) + + |> Chart.toBase64JPGString() + |> fun f -> File.WriteAllText(@"C:\Users\Kevin\source\repos\plotly\Plotly.NET\tests\Plotly.NET.Tests\data\testBase64JPG.txt", f) + +let pngString = + Chart.Point([1.,1.]) + |> Chart.toBase64PNGString() + |> fun f -> File.WriteAllText(@"C:\Users\Kevin\source\repos\plotly\Plotly.NET\tests\Plotly.NET.Tests\data\testBase64PNG.txt", f) + +let svgString = + Chart.Point([1.,1.]) + |> Chart.toSVGString() + |> fun f -> File.WriteAllText(@"C:\Users\Kevin\source\repos\plotly\Plotly.NET\tests\Plotly.NET.Tests\data\testSVGURI.txt", f) diff --git a/src/Plotly.NET.ImageExport/Plotly.NET.ImageExport.fsproj b/src/Plotly.NET.ImageExport/Plotly.NET.ImageExport.fsproj new file mode 100644 index 000000000..248a3dcba --- /dev/null +++ b/src/Plotly.NET.ImageExport/Plotly.NET.ImageExport.fsproj @@ -0,0 +1,43 @@ + + + + net5.0; netstandard2.0 + true + 3390;$(WarnOn) + + + + Kevin Schneider, F# open source contributors + An easily extensible library to render static images from Plotly.NET charts. + An easily extensible library to render static images from Plotly.NET charts. + MIT + https://plotly.net + logo.png + visualization charting plotly fsharp csharp jpg png svg image + https://github.com/plotly/Plotly.NET/ + git + https://github.com/plotly/Plotly.NET/blob/dev/LICENSE + https://github.com/plotly/Plotly.NET/blob/dev/RELEASE_NOTES.md + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Plotly.NET.ImageExport/PuppeteerSharpRenderer.fs b/src/Plotly.NET.ImageExport/PuppeteerSharpRenderer.fs new file mode 100644 index 000000000..426049204 --- /dev/null +++ b/src/Plotly.NET.ImageExport/PuppeteerSharpRenderer.fs @@ -0,0 +1,195 @@ +namespace Plotly.NET.ImageExport + +open Plotly.NET +open PuppeteerSharp + +open System +open System.IO +open System.Text +open System.Text.RegularExpressions + +type PuppeteerSharpRenderer() = + + /// adapted from the original C# implementation by @ilyalatt : https://github.com/ilyalatt/Plotly.NET.PuppeteerRenderer + /// + /// creates a full screen html site for the given chart + let toFullScreenHtml (gChart: GenericChart.GenericChart) = + + gChart + |> GenericChart.mapConfig(fun c -> + DynObj.setValue c "responsive" true + c + ) + |> GenericChart.mapLayout(fun l -> + DynObj.setValue l "width" "100%" + DynObj.setValue l "height" "100%" + l + ) + |> GenericChart.toChartHTML + |> fun html -> html.Replace("width: 600px; height: 600px;", "width: 100%; height: 100%;") + + /// adapted from the original C# implementation by @ilyalatt : https://github.com/ilyalatt/Plotly.NET.PuppeteerRenderer + /// + /// adds the necessary js function calls to render an image with plotly.js + let patchHtml width height (format: StyleParam.ImageFormat) html = + let regex = Regex(@"(Plotly\.newPlot\(.+?\))") + let patchedHtml = + regex.Replace( + html, + ( + fun (x:Match) -> + x.Result( + "$1" + + $".then(x => Plotly.toImage(x, {{ format: '{StyleParam.ImageFormat.toString format}', scale: 2, width: {width}, height: {height} }}))" + + ".then(img => window.plotlyImage = img)" + ) + ) + ) + patchedHtml + + /// adapted from the original C# implementation by @ilyalatt : https://github.com/ilyalatt/Plotly.NET.PuppeteerRenderer + /// + /// attempts to render a chart as static image of the given format with the given dimensions from the given html string + let tryRenderAsync (browser:Browser) (width: int) (height: int) (format: StyleParam.ImageFormat) (html: string) = + async { + let! page = browser.NewPageAsync() |> Async.AwaitTask + try + let! _ = page.SetContentAsync(patchHtml width height format html) |> Async.AwaitTask + let! imgHandle = page.WaitForExpressionAsync("window.plotlyImage") |> Async.AwaitTask + let! imgStr = imgHandle.JsonValueAsync() |> Async.AwaitTask + return imgStr + + finally + page.CloseAsync() |> Async.AwaitTask |> Async.RunSynchronously + } + + /// attempts to render a chart as static image of the given format with the given dimensions from the given html string + let tryRender (browser:Browser) (width: int) (height: int) (format: StyleParam.ImageFormat) (html: string) = + tryRenderAsync browser width height format html + |> Async.RunSynchronously + + /// Initalizes headless browser + let fetchAndLaunchBrowserAsync() = + async { + let browserFetcher = BrowserFetcher() + + let! revision = browserFetcher.DownloadAsync() |> Async.AwaitTask + + let launchOptions = LaunchOptions() + launchOptions.ExecutablePath <- revision.ExecutablePath + launchOptions.Timeout <- 60000 + + return! + Puppeteer.LaunchAsync(launchOptions) + |> Async.AwaitTask + } + + /// Initalizes headless browser + let fetchAndLaunchBrowser() = + fetchAndLaunchBrowserAsync() + |> Async.RunSynchronously + + /// skips the data type part of the given URI + let skipDataTypeString (base64:string) = + let imgBase64StartIdx =base64.IndexOf(",", StringComparison.Ordinal) + 1 + base64.Substring(imgBase64StartIdx) + + /// converst a base64 encoded string URI to a byte array + let getBytesFromBase64String (base64:string) = + base64 + |> skipDataTypeString + |> Convert.FromBase64String + + interface IGenericChartRenderer with + + member this.RenderJPGAsync (width:int, height: int, gChart:GenericChart.GenericChart) = + async { + use! browser = fetchAndLaunchBrowserAsync() + + return! + tryRenderAsync + browser + width + height + StyleParam.ImageFormat.JPEG + (gChart |> toFullScreenHtml) + } + + member this.RenderJPG (width:int, height: int, gChart:GenericChart.GenericChart) = + (this :> IGenericChartRenderer).RenderJPGAsync(width, height, gChart) + |> Async.RunSynchronously + + member this.SaveJPGAsync (path:string, width:int, height: int, gChart:GenericChart.GenericChart) = + async { + let! rendered = (this :> IGenericChartRenderer).RenderJPGAsync(width, height, gChart) + return + rendered + |> getBytesFromBase64String + |> fun base64 -> File.WriteAllBytes($"{path}.jpg", base64) + } + + member this.SaveJPG (path:string, width:int, height: int, gChart:GenericChart.GenericChart) = + (this :> IGenericChartRenderer).SaveJPGAsync(path, width, height, gChart) + |> Async.RunSynchronously + + member this.RenderPNGAsync (width:int, height: int, gChart:GenericChart.GenericChart) = + async { + use! browser = fetchAndLaunchBrowserAsync() + + return! + tryRenderAsync + browser + width + height + StyleParam.ImageFormat.PNG + (gChart |> toFullScreenHtml) + } + + member this.RenderPNG (width:int, height: int, gChart:GenericChart.GenericChart) = + (this :> IGenericChartRenderer).RenderPNGAsync(width, height, gChart) + |> Async.RunSynchronously + + member this.SavePNGAsync (path:string, width:int, height: int, gChart:GenericChart.GenericChart) = + async { + let! rendered = (this :> IGenericChartRenderer).RenderPNGAsync(width, height, gChart) + + return + rendered + |> getBytesFromBase64String + |> fun base64 -> File.WriteAllBytes($"{path}.png", base64) + } + + member this.SavePNG (path:string, width:int, height: int, gChart:GenericChart.GenericChart) = + (this :> IGenericChartRenderer).SavePNGAsync(path, width, height, gChart) + |> Async.RunSynchronously + + member this.RenderSVGAsync (width:int, height: int, gChart:GenericChart.GenericChart) = + async { + use! browser = fetchAndLaunchBrowserAsync() + + let! renderedString = + tryRenderAsync + browser + width + height + StyleParam.ImageFormat.SVG + (gChart |> toFullScreenHtml) + return + renderedString + |> fun svg -> System.Uri.UnescapeDataString(svg) + |> skipDataTypeString + } + + member this.RenderSVG (width:int, height: int, gChart:GenericChart.GenericChart) = + (this :> IGenericChartRenderer).RenderSVGAsync(width, height, gChart) + |> Async.RunSynchronously + + member this.SaveSVGAsync(path:string, width:int, height: int, gChart:GenericChart.GenericChart) = + async { + let! rendered = (this :> IGenericChartRenderer).RenderSVGAsync(width, height, gChart) + return rendered |> fun svg -> File.WriteAllText($"{path}.svg", svg) + } + + member this.SaveSVG (path:string, width:int, height: int, gChart:GenericChart.GenericChart) = + (this :> IGenericChartRenderer).SaveSVGAsync(path, width, height, gChart) + |> Async.RunSynchronously \ No newline at end of file diff --git a/src/Plotly.NET.Interactive/ExtensionVisualTest.ipynb b/src/Plotly.NET.Interactive/ExtensionVisualTest.ipynb index fb7d3ebd9..26e3a5ece 100644 --- a/src/Plotly.NET.Interactive/ExtensionVisualTest.ipynb +++ b/src/Plotly.NET.Interactive/ExtensionVisualTest.ipynb @@ -3,15 +3,9 @@ { "cell_type": "code", "execution_count": null, - "metadata": { - "dotnet_interactive": { - "language": "pwsh" - } - }, - "outputs": [], "source": [ "# notebook paths should be absolute - change this for your setup\r\n", - "$projectRootDir = \"/path/to/your/project/Plotly.NET\"\r\n", + "$projectRootDir = \"C:/Users/Kevin/source/repos/plotly/Plotly.NET\"\r\n", "cd $projectRootDir\r\n", "\r\n", "dotnet tool restore\r\n", @@ -19,74 +13,1506 @@ "\r\n", "# changing versions here if you try out a new version helps you to bust the nuget cache\r\n", "# you still have to restart your notebook kernel, though, if you already loaded a package\r\n", - "dotnet pack /p:PackageVersion=2.0.0-dev4 -o \"$projectRootDir/pkg\"\r\n", + "dotnet pack /p:PackageVersion=2.0.0-dev5 -o \"$projectRootDir/pkg\"\r\n", "\r\n", "ls \"$projectRootDir/pkg\"" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "dotnet_interactive": { - "language": "fsharp" - } - }, + ], "outputs": [ { "data": { - "text/html": "
Restore sources
  • C:/Users/grego/source/repos/Plotly.NET/pkg
" + "text/plain": [ + "Tool 'fake-cli' (version '5.20.4') was restored. Available commands: fake\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Tool 'paket' (version '5.257.0') was restored. Available commands: paket\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Tool 'fsharp.formatting.commandtool' (version '11.1.0') was restored. Available commands: fsdocs\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Restore was successful.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "The last restore is still up to date. Nothing left to do.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "run CopyBinaries\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Building project with version: LocalBuild\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Shortened DependencyGraph for Target CopyBinaries:\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "<== CopyBinaries\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " <== Clean\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " <== Build\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "The running order is:\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Group - 1\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Clean\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Group - 2\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Build\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Group - 3\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - CopyBinaries\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Starting target 'Clean'\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Finished (Success) 'Clean' in 00:00:00.1997738\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Starting target 'Build'\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Starting task 'DotNet:build': Plotly.NET.sln\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET> \"dotnet.exe\" msbuild /version /nologo (In: false, Out: true, Err: true)\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "16.10.1.27001\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET> \"dotnet.exe\" build Plotly.NET.sln --configuration Release /nodeReuse:False \"/bl:C:\\Users\\Kevin\\AppData\\Local\\Temp\\tmpBE44.tmp.binlog\" (In: false, Out: false, Err: false)\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Microsoft (R) Build Engine version 16.10.1+2fd48ab73 for .NET\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Copyright (C) Microsoft Corporation. All rights reserved.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Program Files\\dotnet\\sdk\\5.0.301\\MSBuild.dll -consoleloggerparameters:Summary -distributedlogger:Microsoft.DotNet.Tools.MSBuild.MSBuildLogger,C:\\Program Files\\dotnet\\sdk\\5.0.301\\dotnet.dll*Microsoft.DotNet.Tools.MSBuild.MSBuildForwardingLogger,C:\\Program Files\\dotnet\\sdk\\5.0.301\\dotnet.dll -maxcpucount -property:Configuration=Release -restore -verbosity:m /bl:C:\\Users\\Kevin\\AppData\\Local\\Temp\\tmpBE44.tmp.binlog /nodeReuse:False Plotly.NET.sln\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Determining projects to restore...\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj (in 310 ms).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.ImageExport\\Plotly.NET.ImageExport.fsproj (in 745 ms).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.Interactive\\Plotly.NET.Interactive.fsproj (in 815 ms).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\tests\\Plotly.NET.Tests\\Plotly.NET.Tests.fsproj (in 1.45 sec).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\bin\\Release\\netstandard2.0\\Plotly.NET.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\bin\\Release\\net5.0\\Plotly.NET.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET.Interactive -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.Interactive\\bin\\Release\\net5.0\\Plotly.NET.Interactive.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET.ImageExport -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.ImageExport\\bin\\Release\\net5.0\\Plotly.NET.ImageExport.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET.Tests -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\tests\\Plotly.NET.Tests\\bin\\Release\\net5.0\\Plotly.NET.Tests.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Build succeeded.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " 16 Warning(s)\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " 0 Error(s)\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Time Elapsed 00:00:33.10\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Fsc: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Fsc: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Fsc: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Fsc: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Fsc: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Fsc: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Finished (Success) 'DotNet:build' in 00:00:35.1270633\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Finished (Success) 'Build' in 00:00:35.1342112\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Starting target 'CopyBinaries'\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "(\"C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\bin\\Release\",\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " \"bin\\Plotly.NET\")\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "(\"C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.ImageExport\\bin\\Release\",\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " \"bin\\Plotly.NET.ImageExport\")\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "(\"C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.Interactive\\bin\\Release\",\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " \"bin\\Plotly.NET.Interactive\")\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Finished (Success) 'CopyBinaries' in 00:00:00.4017069\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "---------------------------------------------------------------------\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Build Time Report\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "---------------------------------------------------------------------\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Target Duration\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "------ --------\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Clean 00:00:00.1951835\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Build 00:00:35.1340651\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "CopyBinaries 00:00:00.4015674\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Total: 00:00:35.8823875\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Status: Ok\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "---------------------------------------------------------------------\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\build.fsx (141,12)-(141,26): FSharpErrorSeverity.Warning FS1182: The value 'standardParams' is unused\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Performance:\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Cli parsing: 322 milliseconds\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Packages: 94 milliseconds\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Script analyzing: 36 milliseconds\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Script running: 36 seconds\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Script cleanup: 23 milliseconds\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " - Runtime: 37 seconds\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Microsoft (R) Build Engine version 16.10.1+2fd48ab73 for .NET\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Copyright (C) Microsoft Corporation. All rights reserved.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Determining projects to restore...\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj (in 390 ms).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.ImageExport\\Plotly.NET.ImageExport.fsproj (in 593 ms).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.Interactive\\Plotly.NET.Interactive.fsproj (in 700 ms).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Restored C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\tests\\Plotly.NET.Tests\\Plotly.NET.Tests.fsproj (in 1.5 sec).\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Chart.fs(616,9): warning FS0044: This construct is deprecated. Use the constructors with the mandatory mode argument for full functionality [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\bin\\Debug\\netstandard2.0\\Plotly.NET.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(32,50): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(39,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(40,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(41,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\CandelstickExtension.fs(42,74): warning FS0044: This construct is deprecated. this type will be removed in favor of the 'StockData' type in 2.0. [C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\Plotly.NET.fsproj]\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET\\bin\\Debug\\net5.0\\Plotly.NET.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Successfully created package 'C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\pkg\\Plotly.NET.2.0.0-dev5.nupkg'.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Successfully created package 'C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\pkg\\Plotly.NET.2.0.0-dev5.snupkg'.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET.Interactive -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.Interactive\\bin\\Debug\\net5.0\\Plotly.NET.Interactive.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Successfully created package 'C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\pkg\\Plotly.NET.Interactive.2.0.0-dev5.nupkg'.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Successfully created package 'C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\pkg\\Plotly.NET.Interactive.2.0.0-dev5.snupkg'.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Plotly.NET.ImageExport -> C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\src\\Plotly.NET.ImageExport\\bin\\Debug\\net5.0\\Plotly.NET.ImageExport.dll\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Successfully created package 'C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\pkg\\Plotly.NET.ImageExport.2.0.0-dev5.nupkg'.\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + " Directory: C:\\Users\\Kevin\\source\\repos\\plotly\\Plotly.NET\\pkg\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "Mode LastWriteTime Length Name\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "---- ------------- ------ ----\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "-a--- 7/9/2021 2:40 PM 856987 Plotly.NET.2.0.0-dev5.nupkg\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/plain": [ + "-a--- 7/9/2021 2:40 PM 83819 Plotly.NET.2.0.0-dev5.snupkg\r\n" + ] }, - "output_type": "unknown" + "metadata": {}, + "output_type": "display_data" }, { "data": { - "text/plain": "Installed package Plotly.NET.Interactive version 2.0.0-dev4" + "text/plain": [ + "-a--- 7/9/2021 2:40 PM 51671 Plotly.NET.ImageExport.2.0.0-dev5.nupkg\r\n" + ] }, - "output_type": "unknown" + "metadata": {}, + "output_type": "display_data" }, { "data": { - "text/plain": "Installed package Plotly.NET version 2.0.0-dev4" + "text/plain": [ + "-a--- 7/9/2021 2:40 PM 41265 Plotly.NET.Interactive.2.0.0-dev5.nupkg\r\n" + ] }, - "output_type": "unknown" + "metadata": {}, + "output_type": "display_data" }, { "data": { - "text/markdown": "Loading extensions from `Plotly.NET.Interactive.dll`" + "text/plain": [ + "-a--- 7/9/2021 2:40 PM 2684 Plotly.NET.Interactive.2.0.0-dev5.snupkg\r\n" + ] }, - "output_type": "unknown" + "metadata": {}, + "output_type": "display_data" }, { "data": { - "text/markdown": "Added Kernel Extension including formatters for GenericChart" + "text/plain": [ + "\r\n" + ] }, - "output_type": "unknown" + "metadata": {}, + "output_type": "display_data" } ], - "source": [ - "// be advised, that you always should set absolute paths for local nuget packages - change this to reflect your own setup\r\n", - "#i \"nuget: /path/to/your/project/Plotly.NET/pkg\"\r\n", - "#r \"nuget: Plotly.NET, 2.0.0-dev4\"\r\n", - "#r \"nuget: Plotly.NET.Interactive, 2.0.0-dev4\"" - ] + "metadata": { + "dotnet_interactive": { + "language": "pwsh" + } + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "dotnet_interactive": { - "language": "fsharp" - } - }, + "source": [ + "// be advised, that you always should set absolute paths for local nuget packages - change this to reflect your own setup\r\n", + "#i \"nuget: C:/Users/Kevin/source/repos/plotly/Plotly.NET/pkg\"\r\n", + "#r \"nuget: Plotly.NET, 2.0.0-dev5\"\r\n", + "#r \"nuget: Plotly.NET.ImageExport, 2.0.0-dev5\"\r\n", + "#r \"nuget: Plotly.NET.Interactive, 2.0.0-dev5\"" + ], "outputs": [ { "data": { - "text/html": "
\r\n\r\n" + "text/html": [ + "
Restore sources
  • C:/Users/Kevin/source/repos/plotly/Plotly.NET/pkg
Installed Packages
  • fsharp.core, 5.0.0
  • humanizer.core, 2.2.0
  • microsoft.aspnetcore.html.abstractions, 2.2.0
  • microsoft.aspnetcore.webutilities, 2.0.2
  • microsoft.bcl.asyncinterfaces, 5.0.0
  • microsoft.codeanalysis.common, 3.9.0
  • microsoft.codeanalysis.csharp, 3.9.0
  • microsoft.codeanalysis.csharp.workspaces, 3.9.0
  • microsoft.codeanalysis.visualbasic, 3.9.0
  • microsoft.codeanalysis.visualbasic.workspaces, 3.9.0
  • microsoft.codeanalysis.workspaces.common, 3.9.0
  • microsoft.dotnet.interactive, 1.0.0-beta.21330.2
  • microsoft.dotnet.interactive.formatting, 1.0.0-beta.21330.2
  • microsoft.extensions.dependencyinjection.abstractions, 2.0.0
  • microsoft.extensions.logging, 2.0.2
  • microsoft.extensions.logging.abstractions, 2.0.2
  • microsoft.extensions.options, 2.0.2
  • microsoft.extensions.primitives, 2.0.0
  • microsoft.net.http.headers, 2.0.2
  • newtonsoft.json, 12.0.3
  • nuget.frameworks, 5.0.0
  • Plotly.NET, 2.0.0-dev5
  • Plotly.NET.ImageExport, 2.0.0-dev5
  • Plotly.NET.Interactive, 2.0.0-dev5
  • puppeteersharp, 4.0.0
  • sharpziplib, 1.3.1
  • system.commandline, 2.0.0-beta1.21308.1
  • system.commandline.rendering, 0.3.0-alpha.20427.1
  • system.composition.attributedmodel, 1.0.31
  • system.composition.convention, 1.0.31
  • system.composition.hosting, 1.0.31
  • system.composition.runtime, 1.0.31
  • system.composition.typedparts, 1.0.31
  • system.io.pipelines, 5.0.0
  • system.reactive, 5.0.0
  • system.text.encodings.web, 5.0.1
  • system.text.json, 5.0.2
" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/markdown": [ + "Loading extensions from `Plotly.NET.Interactive.dll`" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/markdown": [ + "Added Kernel Extension including formatters for Plotly.NET charts." + ] }, - "output_type": "unknown" + "metadata": {}, + "output_type": "display_data" } ], + "metadata": { + "dotnet_interactive": { + "language": "fsharp" + } + } + }, + { + "cell_type": "code", + "execution_count": null, "source": [ "open Plotly.NET\r\n", "\r\n", @@ -95,7 +1521,112 @@ "\r\n", "List.zip xs ys\r\n", "|> Chart.Point" - ] + ], + "outputs": [ + { + "data": { + "text/html": [ + "
\r\n", + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "metadata": { + "dotnet_interactive": { + "language": "fsharp" + } + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "open Plotly.NET\r\n", + "open Plotly.NET.ImageExport\r\n", + "\r\n", + "let chartJPEG = \r\n", + " Chart.Point([1.,2.])\r\n", + " |> Chart.toBase64JPGString(Width=400, Height=400)\r\n", + "\r\n", + "let chartPNG = \r\n", + " Chart.Point([1.,2.])\r\n", + " |> Chart.toBase64PNGString(Width=400, Height=400)\r\n", + "\r\n", + "let chartSVG = \r\n", + " Chart.Point([1.,2.])\r\n", + " |> Chart.toSVGString(Width=400, Height=400)\r\n", + "\r\n", + "let imageHtml chartImage = $\"\"\"\r\n", + "\"\"\"\r\n", + "\r\n", + "DisplayExtensions.DisplayAs(imageHtml chartJPEG,\"text/html\")\r\n", + "DisplayExtensions.DisplayAs(imageHtml chartPNG,\"text/html\")\r\n", + "DisplayExtensions.DisplayAs(chartSVG,\"image/svg+xml\")\r\n" + ], + "outputs": [ + { + "data": { + "text/html": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "\r\n" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/svg+xml": "00.511.5211.522.53" + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "metadata": { + "dotnet_interactive": { + "language": "fsharp" + } + } } ], "metadata": { diff --git a/src/Plotly.NET.Interactive/Untitled-1.html b/src/Plotly.NET.Interactive/Untitled-1.html new file mode 100644 index 000000000..3117e262b --- /dev/null +++ b/src/Plotly.NET.Interactive/Untitled-1.html @@ -0,0 +1,243 @@ + \ No newline at end of file diff --git a/src/Plotly.NET/GenericChart.fs b/src/Plotly.NET/GenericChart.fs index a6294be79..8b6e9edf7 100644 --- a/src/Plotly.NET/GenericChart.fs +++ b/src/Plotly.NET/GenericChart.fs @@ -392,8 +392,15 @@ module GenericChart = let mapLayout f gChart = match gChart with | Chart (trace, layout, config, displayOpts) -> Chart (trace,f layout,config, displayOpts) - | MultiChart (traces, layout, config, displayOpts) -> MultiChart (traces,f layout,config, displayOpts) /// + | MultiChart (traces, layout, config, displayOpts) -> MultiChart (traces,f layout,config, displayOpts) + + /// + let mapConfig f gChart = + match gChart with + | Chart (trace, layout, config, displayOpts) -> Chart (trace, layout, f config, displayOpts) + | MultiChart (traces, layout, config, displayOpts) -> MultiChart (traces, layout, f config, displayOpts) + /// let mapDisplayOptions f gChart = match gChart with | Chart (trace, layout, config, displayOpts) -> Chart (trace, layout, config, f displayOpts) diff --git a/src/Plotly.NET/Plotly.NET.fsproj b/src/Plotly.NET/Plotly.NET.fsproj index 30f3a8f20..5848525ee 100644 --- a/src/Plotly.NET/Plotly.NET.fsproj +++ b/src/Plotly.NET/Plotly.NET.fsproj @@ -84,7 +84,6 @@ - diff --git a/src/Plotly.NET/TestScript.fsx b/src/Plotly.NET/TestScript.fsx deleted file mode 100644 index 496b1c612..000000000 --- a/src/Plotly.NET/TestScript.fsx +++ /dev/null @@ -1,69 +0,0 @@ -//#I "./bin/Debug" -//#r "./bin/Debug/netstandard2.0/Newtonsoft.Json.dll" -#r @"..\..\bin\Plotly.NET\netstandard2.0\Plotly.NET.dll" -#r "netstandard" - -open Plotly.NET - - -let x = seq [1.; 2.; 3.; 4.; 5.; 6.; 7.; 8.; 9.; 10.; ] // 9.; 8.; 7.; 6.; 5.; 4.; 3.; 2.; 1.] -let y = seq [5.; 2.5; 5.; 7.5; 5.; 2.5; 7.5; 4.5; 5.5; 5.] -let y' = seq [2.; 1.5; 5.; 1.5; 3.; 2.5; 2.5; 1.5; 3.5; 1.] - -Chart.Spline(x,y',Name="spline") -|> Chart.withConfig - (Config.init - ( - Autosizable = true, - ShowEditInChartStudio = true, - ToImageButtonOptions = - ToImageButtonOptions.init - ( - Format = StyleParam.ImageFormat.SVG, - Filename = "SOOOOS" - ) - ) - ) -|> Chart.Show -//|> Chart.withYError(Options.Error(Array=[1.; 2.; 3.; 4.; 5.; 6.; 7.; 8.; 9.; 10.; ])) - - -//|> Chart.withLineStyle(Width=2,Dash=StyleParam.DrawingStyle.Dot) -//|> Chart.withLineOption(Options.Line(Width=10)) -//|> Chart.withX_AxisStyle("x axis title") -//|> Chart.withY_AxisStyle("y axis title") -//|> layoutJson -//|> GenericChart.toChartHtmlWithSize 500 500 - -//|> Chart.ShowAsImage StyleParam.ImageFormat.SVG - -let dims' = - [ - Dimensions.init(["Cat1";"Cat1";"Cat1";"Cat1";"Cat2";"Cat2";"Cat3"],Label="A") - Dimensions.init([0;1;0;1;0;0;0],Label="B",TickText=["YES","NO"]) - ] - -Chart.ParallelCategories(dims=dims',Color=[0.;1.;0.;1.;0.;0.;0.],Colorscale = StyleParam.Colorscale.Blackbody) -|> Chart.Show - -let data3d = List.zip3 [0 .. 15] [0 .. 15] [0 .. 15] -let data2d = List.zip [0 .. 15] [0 .. 15] - -[ - Chart.Point (data2d) - Chart.Scatter3d(xyz=data3d,mode = StyleParam.Mode.Markers) - Chart.Scatter3d(xyz=data3d,mode = StyleParam.Mode.Markers) - Chart.Scatter3d(xyz=data3d,mode = StyleParam.Mode.Markers) - Chart.Scatter3d(xyz=data3d,mode = StyleParam.Mode.Markers) -] -|> Chart.Stack 2 -|> Chart.Show - - - - - - - - - diff --git a/tests/Plotly.NET.Tests/BasicHtmlGeneration.fs b/tests/Plotly.NET.Tests/BasicHtmlGeneration.fs index cae1ce2b7..f1096b334 100644 --- a/tests/Plotly.NET.Tests/BasicHtmlGeneration.fs +++ b/tests/Plotly.NET.Tests/BasicHtmlGeneration.fs @@ -1,4 +1,4 @@ -module Tests +module Tests.Core open Expecto open Plotly.NET diff --git a/tests/Plotly.NET.Tests/ImageExport.fs b/tests/Plotly.NET.Tests/ImageExport.fs new file mode 100644 index 000000000..731d6d7b4 --- /dev/null +++ b/tests/Plotly.NET.Tests/ImageExport.fs @@ -0,0 +1,34 @@ +module Tests.ImageExport + +open Expecto +open Plotly.NET +open Plotly.NET.ImageExport + +open System.IO + +let testBase64JPG = File.ReadAllText (__SOURCE_DIRECTORY__ + "/data/testBase64JPG.txt") +let testBase64PNG = File.ReadAllText (__SOURCE_DIRECTORY__ + "/data/testBase64PNG.txt") + +let testChart = Chart.Point([1.,1.]) + +//[] +//let jpgTests = +// testList "ImageExport_JPG" [ +// testCase "Chart.toBase64JPGString should create valid base64 JPG string" ( fun () -> +// Expect.equal +// (testChart |> Chart.toBase64JPGString()) +// testBase64JPG +// "Nope" +// ) +// ] + +//[] +//let pngTests = +// testList "ImageExport_PNG" [ +// testCase "Chart.toBase64PNGString should create valid base64 PNG string" ( fun () -> +// Expect.equal +// (testChart |> Chart.toBase64PNGString()) +// testBase64PNG +// "Nope" +// ) +// ] \ No newline at end of file diff --git a/tests/Plotly.NET.Tests/Main.fs b/tests/Plotly.NET.Tests/Main.fs index 11e78d2f3..1f34e91d4 100644 --- a/tests/Plotly.NET.Tests/Main.fs +++ b/tests/Plotly.NET.Tests/Main.fs @@ -3,4 +3,4 @@ open Expecto [] let main argv = - Tests.runTestsInAssembly defaultConfig argv + Tests.runTestsInAssemblyWithCLIArgs [] argv diff --git a/tests/Plotly.NET.Tests/Plotly.NET.Tests.fsproj b/tests/Plotly.NET.Tests/Plotly.NET.Tests.fsproj index 34f460526..2df61741f 100644 --- a/tests/Plotly.NET.Tests/Plotly.NET.Tests.fsproj +++ b/tests/Plotly.NET.Tests/Plotly.NET.Tests.fsproj @@ -7,6 +7,7 @@ + @@ -19,5 +20,6 @@ + diff --git a/tests/Plotly.NET.Tests/data/testBase64JPG.txt b/tests/Plotly.NET.Tests/data/testBase64JPG.txt new file mode 100644 index 000000000..6b0561973 --- /dev/null +++ b/tests/Plotly.NET.Tests/data/testBase64JPG.txt @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/tests/Plotly.NET.Tests/data/testBase64PNG.txt b/tests/Plotly.NET.Tests/data/testBase64PNG.txt new file mode 100644 index 000000000..250a6560b --- /dev/null +++ b/tests/Plotly.NET.Tests/data/testBase64PNG.txt @@ -0,0 +1 @@ + \ No newline at end of file