mirror of
https://github.com/spectreconsole/spectre.console.git
synced 2025-10-25 15:19:23 +00:00
Compare commits
16 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e280b82679 | ||
|
|
6932c95731 | ||
|
|
ee305702e8 | ||
|
|
63abcc92ba | ||
|
|
acf01e056f | ||
|
|
501db5d287 | ||
|
|
cbed41e637 | ||
|
|
3c504155bc | ||
|
|
ae32785f21 | ||
|
|
c61e386440 | ||
|
|
b7cd7dd53e | ||
|
|
3e1251b86a | ||
|
|
01fdbac51e | ||
|
|
b0b988a1e7 | ||
|
|
2a9fa223de | ||
|
|
4f6eca4fcb |
28
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
28
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title: ''
|
||||||
|
labels: bug
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Information**
|
||||||
|
- OS: [eg Windows/Linux/MacOS]
|
||||||
|
- Version: [e.g. 0.33.0]
|
||||||
|
- Terminal: [e.g Windows Terminal]
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
A clear and concise description of what the bug is.
|
||||||
|
|
||||||
|
**To Reproduce**
|
||||||
|
Steps to reproduce the behavior.
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Screenshots**
|
||||||
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context about the problem here.
|
||||||
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
---
|
||||||
|
name: Feature request
|
||||||
|
about: Suggest an idea for this project
|
||||||
|
title: ''
|
||||||
|
labels: enhancement
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||||
|
|
||||||
|
**Describe the solution you'd like**
|
||||||
|
A clear and concise description of what you want to happen.
|
||||||
|
|
||||||
|
**Describe alternatives you've considered**
|
||||||
|
A clear and concise description of any alternative solutions or features you've considered.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context or screenshots about the feature request here.
|
||||||
9
.github/workflows/ci.yaml
vendored
9
.github/workflows/ci.yaml
vendored
@@ -69,14 +69,7 @@ jobs:
|
|||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
dotnet tool restore
|
dotnet tool restore
|
||||||
dotnet example info
|
dotnet example --all
|
||||||
dotnet example tables
|
|
||||||
dotnet example grids
|
|
||||||
dotnet example panels
|
|
||||||
dotnet example colors
|
|
||||||
dotnet example emojis
|
|
||||||
dotnet example exceptions
|
|
||||||
dotnet example calendars
|
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
shell: bash
|
shell: bash
|
||||||
|
|||||||
110
README.md
110
README.md
@@ -2,19 +2,17 @@
|
|||||||
|
|
||||||
_[](https://www.nuget.org/packages/spectre.console)_
|
_[](https://www.nuget.org/packages/spectre.console)_
|
||||||
|
|
||||||
A .NET Standard 2.0 library that makes it easier to create beautiful console applications.
|
A .NET 5/.NET Standard 2.0 library that makes it easier to create beautiful, cross platform, console applications.
|
||||||
It is heavily inspired by the excellent [Rich library](https://github.com/willmcgugan/rich)
|
It is heavily inspired by the excellent [Rich library](https://github.com/willmcgugan/rich)
|
||||||
for Python.
|
for Python.
|
||||||
|
|
||||||
## Table of Contents
|
## Table of Contents
|
||||||
|
|
||||||
1. [Features](#features)
|
1. [Features](#features)
|
||||||
2. [Example](#example)
|
2. [Installing](#installing)
|
||||||
3. [Installing](#installing)
|
3. [Documentation](#documentation)
|
||||||
4. [Usage](#usage)
|
4. [Examples](#examples)
|
||||||
4.1. [Using the static API](#using-the-static-api)
|
5. [License](#license)
|
||||||
4.2. [Creating a console](#creating-a-console)
|
|
||||||
5. [Running examples](#running-examples)
|
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
@@ -25,77 +23,27 @@ for Python.
|
|||||||
and blinking text.
|
and blinking text.
|
||||||
* Supports 3/4/8/24-bit colors in the terminal.
|
* Supports 3/4/8/24-bit colors in the terminal.
|
||||||
The library will detect the capabilities of the current terminal
|
The library will detect the capabilities of the current terminal
|
||||||
and downgrade colors as needed.
|
and downgrade colors as needed.
|
||||||
|
|
||||||
## Example
|
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
## Installing
|
## Installing
|
||||||
|
|
||||||
The fastest way of getting started using Spectre.Console is to install the NuGet package.
|
The fastest way of getting started using `Spectre.Console` is to install the NuGet package.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
dotnet add package Spectre.Console
|
dotnet add package Spectre.Console
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Documentation
|
||||||
|
|
||||||
The `Spectre.Console` API is stateful and is not thread-safe.
|
The documentation for `Spectre.Console` can be found at
|
||||||
If you need to write to the console from different threads, make sure that
|
https://spectresystems.github.io/spectre.console/
|
||||||
you take appropriate precautions, just like when you use the
|
|
||||||
regular `System.Console` API.
|
|
||||||
|
|
||||||
If the current terminal does not support ANSI escape sequences,
|
## Examples
|
||||||
`Spectre.Console` will fallback to using the `System.Console` API.
|
|
||||||
|
|
||||||
_NOTE: This library is currently under development and APIs
|
To see `Spectre.Console` in action, install the
|
||||||
might change or get removed at any point up until a 1.0 release._
|
|
||||||
|
|
||||||
### Using the static API
|
|
||||||
|
|
||||||
The static API is perfect when you just want to output text
|
|
||||||
like you usually do with the `System.Console` API, but prettier.
|
|
||||||
|
|
||||||
```csharp
|
|
||||||
AnsiConsole.Foreground = Color.CornflowerBlue;
|
|
||||||
AnsiConsole.Decoration = Decoration.Underline | Decoration.Bold;
|
|
||||||
AnsiConsole.WriteLine("Hello World!");
|
|
||||||
|
|
||||||
AnsiConsole.Reset();
|
|
||||||
AnsiConsole.MarkupLine("[bold yellow on red]{0}[/] [underline]world[/]!", "Goodbye");
|
|
||||||
```
|
|
||||||
|
|
||||||
If you want to get a reference to the default `IAnsiConsole`,
|
|
||||||
you can access it via `AnsiConsole.Console`.
|
|
||||||
|
|
||||||
### Creating a console
|
|
||||||
|
|
||||||
Sometimes it's useful to explicitly create a console with specific
|
|
||||||
capabilities, such as during unit testing when you want control
|
|
||||||
over the environment your code runs in.
|
|
||||||
|
|
||||||
It's recommended to not use `AnsiConsole` in code that run as
|
|
||||||
part of a unit test.
|
|
||||||
|
|
||||||
```csharp
|
|
||||||
IAnsiConsole console = AnsiConsole.Create(
|
|
||||||
new AnsiConsoleSettings()
|
|
||||||
{
|
|
||||||
Ansi = AnsiSupport.Yes,
|
|
||||||
ColorSystem = ColorSystemSupport.TrueColor,
|
|
||||||
Out = new StringWriter(),
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
_NOTE: Even if you can specify a specific color system to use
|
|
||||||
when manually creating a console, remember that the user's terminal
|
|
||||||
might not be able to use it, so unless you're creating an IAnsiConsole
|
|
||||||
for testing, always use `ColorSystemSupport.Detect` and `AnsiSupport.Detect`._
|
|
||||||
|
|
||||||
## Running examples
|
|
||||||
|
|
||||||
To see Spectre.Console in action, install the
|
|
||||||
[dotnet-example](https://github.com/patriksvensson/dotnet-example)
|
[dotnet-example](https://github.com/patriksvensson/dotnet-example)
|
||||||
global tool.
|
global tool.
|
||||||
|
|
||||||
@@ -107,34 +55,18 @@ Now you can list available examples in this repository:
|
|||||||
|
|
||||||
```
|
```
|
||||||
> dotnet example
|
> dotnet example
|
||||||
|
|
||||||
╭────────────┬───────────────────────────────────────┬──────────────────────────────────────────────────────╮
|
|
||||||
│ Name │ Path │ Description │
|
|
||||||
├────────────┼───────────────────────────────────────┼──────────────────────────────────────────────────────┤
|
|
||||||
│ Borders │ examples/Borders/Borders.csproj │ Demonstrates the different kind of borders. │
|
|
||||||
│ Calendars │ examples/Calendars/Calendars.csproj │ Demonstrates how to render calendars. │
|
|
||||||
│ Colors │ examples/Colors/Colors.csproj │ Demonstrates how to use colors in the console. │
|
|
||||||
│ Columns │ examples/Columns/Columns.csproj │ Demonstrates how to render data into columns. │
|
|
||||||
│ Emojis │ examples/Emojis/Emojis.csproj │ Demonstrates how to render emojis. │
|
|
||||||
│ Exceptions │ examples/Exceptions/Exceptions.csproj │ Demonstrates how to render formatted exceptions. │
|
|
||||||
│ Grids │ examples/Grids/Grids.csproj │ Demonstrates how to render grids in a console. │
|
|
||||||
│ Info │ examples/Info/Info.csproj │ Displays the capabilities of the current console. │
|
|
||||||
│ Links │ examples/Links/Links.csproj │ Demonstrates how to render links in a console. │
|
|
||||||
│ Panels │ examples/Panels/Panels.csproj │ Demonstrates how to render items in panels. │
|
|
||||||
│ Rules │ examples/Rules/Rules.csproj │ Demonstrates how to render horizontal rules (lines). │
|
|
||||||
│ Tables │ examples/Tables/Tables.csproj │ Demonstrates how to render tables in a console. │
|
|
||||||
╰────────────┴───────────────────────────────────────┴──────────────────────────────────────────────────────╯
|
|
||||||
```
|
```
|
||||||
|
|
||||||
And to run an example:
|
And to run an example:
|
||||||
|
|
||||||
```
|
```
|
||||||
> dotnet example tables
|
> dotnet example tables
|
||||||
┌──────────┬──────────┬────────┐
|
```
|
||||||
│ Foo │ Bar │ Baz │
|
|
||||||
├──────────┼──────────┼────────┤
|
## License
|
||||||
│ Hello │ World! │ │
|
|
||||||
│ Bonjour │ le │ monde! │
|
Copyright © Spectre Systems.
|
||||||
│ Hej │ Världen! │ │
|
|
||||||
└──────────┴──────────┴────────┘
|
Spectre.Console is provided as-is under the MIT license. For more information see LICENSE.
|
||||||
```
|
|
||||||
|
* For SixLabors.ImageSharp, see https://github.com/SixLabors/ImageSharp/blob/master/LICENSE
|
||||||
65
README.zh.md
Normal file
65
README.zh.md
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
# `Spectre.Console`
|
||||||
|
|
||||||
|
_[](https://www.nuget.org/packages/spectre.console)_
|
||||||
|
|
||||||
|
`Spectre.Console`是一个 .NET 5/.NET Standard 2.0 的库,能让您在终端里更方便地生成精美的界面。
|
||||||
|
|
||||||
|
深受 [Rich](https://github.com/willmcgugan/rich) 这个优秀库的启发。
|
||||||
|
|
||||||
|
## 目录
|
||||||
|
|
||||||
|
1. [功能](#features)
|
||||||
|
2. [安装](#installing)
|
||||||
|
3. [文档](#documentation)
|
||||||
|
4. [例子](#examples)
|
||||||
|
5. [License](#license)
|
||||||
|
|
||||||
|
## 功能
|
||||||
|
|
||||||
|
* 编写时考虑到了单元测试。
|
||||||
|
* 支持 tables、grid、panel 和 [rich](https://github.com/willmcgugan/rich) 所支持的标记语言。
|
||||||
|
* 支持大部分的 SRG 参数,包括粗体、暗淡字、斜体、下划线、删除线和闪烁文本。
|
||||||
|
* 支持终端显示 3/4/8/24 位色。自动检测终端类型,自适应颜色范围。
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## 安装
|
||||||
|
|
||||||
|
最快的安装方式,就是用NuGet包管理直接安装Spectre.Console。
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
dotnet add package Spectre.Console
|
||||||
|
```
|
||||||
|
|
||||||
|
## 文档
|
||||||
|
|
||||||
|
`Spectre.Console`的文档可以在这里查看
|
||||||
|
https://spectresystems.github.io/spectre.console/
|
||||||
|
|
||||||
|
## 例子
|
||||||
|
|
||||||
|
如果想直接运行`Spectre.Console`的例子,则需要安装[dotnet-example](https://github.com/patriksvensson/dotnet-example)工具。
|
||||||
|
|
||||||
|
```
|
||||||
|
> dotnet tool restore
|
||||||
|
```
|
||||||
|
|
||||||
|
然后你可以列出仓库里的所有例子:
|
||||||
|
|
||||||
|
```
|
||||||
|
> dotnet example
|
||||||
|
```
|
||||||
|
|
||||||
|
跑一个看看效果:
|
||||||
|
|
||||||
|
```
|
||||||
|
> dotnet example tables
|
||||||
|
```
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
版权所有 © Spectre Systems。
|
||||||
|
|
||||||
|
Spectre.Console 基于 MIT 协议提供。查看 LICENSE 文件了解更多信息。
|
||||||
|
|
||||||
|
* SixLabors.ImageSharp 的协议请查看 https://github.com/SixLabors/ImageSharp/blob/master/LICENSE
|
||||||
43
docs/input/appendix/spinners.md
Normal file
43
docs/input/appendix/spinners.md
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
Title: Spinners
|
||||||
|
Order: 4
|
||||||
|
---
|
||||||
|
|
||||||
|
For all available spinners, see https://jsfiddle.net/sindresorhus/2eLtsbey/embedded/result/
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
|
||||||
|
Spinners can be used with [Progress](xref:progress) and [Status](xref:status).
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
AnsiConsole.Status()
|
||||||
|
.Spinner(Spinner.Known.Star)
|
||||||
|
.Start("Thinking...", ctx => {
|
||||||
|
// Omitted
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
# Implementing a spinner
|
||||||
|
|
||||||
|
To implement your own spinner, all you have to do is
|
||||||
|
inherit from the `Spinner` base class.
|
||||||
|
|
||||||
|
In the example below, the spinner will alterate between
|
||||||
|
the characters `A`, `B` and `C` every 100 ms.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public sealed class MySpinner : Spinner
|
||||||
|
{
|
||||||
|
// The interval for each frame
|
||||||
|
public override TimeSpan Interval => TimeSpan.FromMilliseconds(100);
|
||||||
|
|
||||||
|
// Whether or not the spinner contains unicode characters
|
||||||
|
public override bool IsUnicode => false;
|
||||||
|
|
||||||
|
// The individual frames of the spinner
|
||||||
|
public override IReadOnlyList<string> Frames =>
|
||||||
|
new List<string>
|
||||||
|
{
|
||||||
|
"A", "B", "C",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
```
|
||||||
BIN
docs/input/assets/images/progress.gif
Normal file
BIN
docs/input/assets/images/progress.gif
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 164 KiB |
BIN
docs/input/assets/images/progress.png
Normal file
BIN
docs/input/assets/images/progress.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 14 KiB |
BIN
docs/input/assets/images/progress_fallback.png
Normal file
BIN
docs/input/assets/images/progress_fallback.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 33 KiB |
BIN
docs/input/assets/images/status.gif
Normal file
BIN
docs/input/assets/images/status.gif
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 257 KiB |
@@ -31,5 +31,4 @@ $(document).ready(function () {
|
|||||||
}; // keyup
|
}; // keyup
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
}); // ready
|
}); // ready
|
||||||
78
docs/input/progress.md
Normal file
78
docs/input/progress.md
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
Title: Progress
|
||||||
|
Order: 5
|
||||||
|
---
|
||||||
|
|
||||||
|
Spectre.Console can display information about long running tasks in the console.
|
||||||
|
|
||||||
|
<img src="assets/images/progress.png" style="max-width: 100%;margin-bottom:20px;">
|
||||||
|
|
||||||
|
If the current terminal isn't considered "interactive", such as when running
|
||||||
|
in a continuous integration system, or the terminal can't display
|
||||||
|
ANSI control sequence, any progress will be displayed in a simpler way.
|
||||||
|
|
||||||
|
<img src="assets/images/progress_fallback.png" style="max-width: 100%;">
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Synchronous
|
||||||
|
AnsiConsole.Progress()
|
||||||
|
.Start(ctx =>
|
||||||
|
{
|
||||||
|
// Define tasks
|
||||||
|
var task1 = ctx.AddTask("[green]Reticulating splines[/]");
|
||||||
|
var task2 = ctx.AddTask("[green]Folding space[/]");
|
||||||
|
|
||||||
|
while(!ctx.IsFinished)
|
||||||
|
{
|
||||||
|
task1.Increment(1.5);
|
||||||
|
task2.Increment(0.5);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
## Asynchronous progress
|
||||||
|
|
||||||
|
If you prefer to use async/await, you can use `StartAsync` instead of `Start`.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Asynchronous
|
||||||
|
await AnsiConsole.Progress()
|
||||||
|
.StartAsync(async ctx =>
|
||||||
|
{
|
||||||
|
// Define tasks
|
||||||
|
var task1 = ctx.AddTask("[green]Reticulating splines[/]");
|
||||||
|
var task2 = ctx.AddTask("[green]Folding space[/]");
|
||||||
|
|
||||||
|
while (!ctx.IsFinished)
|
||||||
|
{
|
||||||
|
// Simulate some work
|
||||||
|
await Task.Delay(250);
|
||||||
|
|
||||||
|
// Increment
|
||||||
|
task1.Increment(1.5);
|
||||||
|
task2.Increment(0.5);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
# Configure
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Asynchronous
|
||||||
|
AnsiConsole.Progress()
|
||||||
|
.AutoRefresh(false) // Turn off auto refresh
|
||||||
|
.AutoClear(false) // Do not remove the task list when done
|
||||||
|
.Columns(new ProgressColumn[]
|
||||||
|
{
|
||||||
|
new TaskDescriptionColumn(), // Task description
|
||||||
|
new ProgressBarColumn(), // Progress bar
|
||||||
|
new PercentageColumn(), // Percentage
|
||||||
|
new RemainingTimeColumn(), // Remaining time
|
||||||
|
new SpinnerColumn(), // Spinner
|
||||||
|
})
|
||||||
|
.Start(ctx =>
|
||||||
|
{
|
||||||
|
// Omitted
|
||||||
|
});
|
||||||
|
```
|
||||||
@@ -5,7 +5,7 @@ Order: 1
|
|||||||
The fastest way of getting started using Spectre.Console is
|
The fastest way of getting started using Spectre.Console is
|
||||||
to install the NuGet package.
|
to install the NuGet package.
|
||||||
|
|
||||||
```shell
|
```text
|
||||||
> dotnet add package Spectre.Console
|
> dotnet add package Spectre.Console
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
60
docs/input/status.md
Normal file
60
docs/input/status.md
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
Title: Status
|
||||||
|
Order: 6
|
||||||
|
---
|
||||||
|
|
||||||
|
Spectre.Console can display information about long running tasks in the console.
|
||||||
|
|
||||||
|
<img src="assets/images/status.gif" style="max-width: 100%;margin-bottom:20px;">
|
||||||
|
|
||||||
|
If the current terminal isn't considered "interactive", such as when running
|
||||||
|
in a continuous integration system, or the terminal can't display
|
||||||
|
ANSI control sequence, any progress will be displayed in a simpler way.
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Synchronous
|
||||||
|
AnsiConsole.Status()
|
||||||
|
.Start("Thinking...", ctx =>
|
||||||
|
{
|
||||||
|
// Simulate some work
|
||||||
|
AnsiConsole.MarkupLine("Doing some work...");
|
||||||
|
Thread.Sleep(1000);
|
||||||
|
|
||||||
|
// Update the status and spinner
|
||||||
|
ctx.Status("Thinking some more");
|
||||||
|
ctx.Spinner(Spinner.Known.Star);
|
||||||
|
ctx.SpinnerStyle(Style.Parse("green"));
|
||||||
|
|
||||||
|
// Simulate some work
|
||||||
|
AnsiConsole.MarkupLine("Doing some more work...");
|
||||||
|
Thread.Sleep(2000);
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
## Asynchronous progress
|
||||||
|
|
||||||
|
If you prefer to use async/await, you can use `StartAsync` instead of `Start`.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Asynchronous
|
||||||
|
await AnsiConsole.Status()
|
||||||
|
.StartAsync("Thinking...", async ctx =>
|
||||||
|
{
|
||||||
|
// Omitted
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
# Configure
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
AnsiConsole.Status()
|
||||||
|
.AutoRefresh(false)
|
||||||
|
.Spinner(Spinner.Known.Star)
|
||||||
|
.SpinnerStyle(Style.Parse("green bold"))
|
||||||
|
.Start("Thinking...", ctx =>
|
||||||
|
{
|
||||||
|
// Omitted
|
||||||
|
ctx.Refresh();
|
||||||
|
});
|
||||||
|
```
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
Title: Calendar
|
Title: Calendar
|
||||||
Order: 4
|
Order: 2
|
||||||
RedirectFrom: calendar
|
RedirectFrom: calendar
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
106
docs/input/widgets/canvas-image.md
Normal file
106
docs/input/widgets/canvas-image.md
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
Title: Canvas Image
|
||||||
|
Order: 5
|
||||||
|
---
|
||||||
|
|
||||||
|
To add [ImageSharp](https://github.com/SixLabors/ImageSharp) superpowers to
|
||||||
|
your console application to draw images, you will need to install
|
||||||
|
the [Spectre.Console.ImageSharp](https://www.nuget.org/packages/Spectre.Console.ImageSharp) NuGet package.
|
||||||
|
|
||||||
|
```text
|
||||||
|
> dotnet add package Spectre.Console.ImageSharp
|
||||||
|
```
|
||||||
|
|
||||||
|
# Loading images
|
||||||
|
|
||||||
|
Once you've added the `Spectre.Console.ImageSharp` NuGet package,
|
||||||
|
you can create a new instance of `CanvasImage` to draw images to the console.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Load an image
|
||||||
|
var image = new CanvasImage("cake.png");
|
||||||
|
|
||||||
|
// Set the max width of the image.
|
||||||
|
// If no max width is set, the image will take
|
||||||
|
// up as much space as there is available.
|
||||||
|
image.MaxWidth(16);
|
||||||
|
|
||||||
|
// Render the image to the console
|
||||||
|
AnsiConsole.Render(image);
|
||||||
|
```
|
||||||
|
|
||||||
|
## Result
|
||||||
|
|
||||||
|
<pre style="font-size:90%;font-family:consolas,'Courier New',monospace;line-height: normal; padding: 0px;background-color: #222222; padding: 20px;">
|
||||||
|
<span> </span><span style="background-color: #542813"> </span><span style="background-color: #572F1B"> </span><span style="background-color: #4E1F09"> </span><span style="background-color: #5B3826"> </span><span style="background-color: #5E3A29"> </span><span style="background-color: #532611"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #562E1B"> </span><span style="background-color: #634737"> </span><span style="background-color: #562E1A"> </span><span style="background-color: #5D4132"> </span><span style="background-color: #6D584B"> </span><span style="background-color: #624332"> </span><span style="background-color: #562B17"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #512714"> </span><span style="background-color: #654E40"> </span><span style="background-color: #705243"> </span><span style="background-color: #745749"> </span><span style="background-color: #6D5B4F"> </span><span style="background-color: #715E52"> </span><span style="background-color: #644636"> </span><span style="background-color: #6A4433"> </span><span style="background-color: #542916"> </span><span style="background-color: #431C0B"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #491E0A"> </span><span style="background-color: #5C3523"> </span><span style="background-color: #695346"> </span><span style="background-color: #705C4F"> </span><span style="background-color: #654838"> </span><span style="background-color: #654A3A"> </span><span style="background-color: #726154"> </span><span style="background-color: #715D50"> </span><span style="background-color: #B8A79F"> </span><span style="background-color: #AE988F"> </span><span style="background-color: #6F4A39"> </span><span style="background-color: #441906"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #532916"> </span><span style="background-color: #8A6C5E"> </span><span style="background-color: #C2B3AB"> </span><span style="background-color: #8B786E"> </span><span style="background-color: #6B584C"> </span><span style="background-color: #695143"> </span><span style="background-color: #6C5648"> </span><span style="background-color: #6F5D51"> </span><span style="background-color: #816A55"> </span><span style="background-color: #E7E1DA"> </span><span style="background-color: #F9F5EE"> </span><span style="background-color: #BAA593"> </span><span style="background-color: #61381F"> </span><span> </span>
|
||||||
|
<span style="background-color: #421C0A"> </span><span style="background-color: #603826"> </span><span style="background-color: #9E8479"> </span><span style="background-color: #E2DAD6"> </span><span style="background-color: #FBF9F6"> </span><span style="background-color: #F0EADF"> </span><span style="background-color: #C4B59D"> </span><span style="background-color: #9D8663"> </span><span style="background-color: #786451"> </span><span style="background-color: #705D4E"> </span><span style="background-color: #BFA052"> </span><span style="background-color: #FEE88B"> </span><span style="background-color: #FDE580"> </span><span style="background-color: #E2C362"> </span><span style="background-color: #794E1D"> </span><span> </span>
|
||||||
|
<span style="background-color: #4B1D05"> </span><span style="background-color: #A6844C"> </span><span style="background-color: #E9D595"> </span><span style="background-color: #F1DC92"> </span><span style="background-color: #F5DD83"> </span><span style="background-color: #FBE278"> </span><span style="background-color: #FFE36E"> </span><span style="background-color: #F1D25E"> </span><span style="background-color: #866F4B"> </span><span style="background-color: #726256"> </span><span style="background-color: #967945"> </span><span style="background-color: #F5D456"> </span><span style="background-color: #F8D756"> </span><span style="background-color: #E1BE4A"> </span><span style="background-color: #7D511B"> </span><span> </span>
|
||||||
|
<span style="background-color: #4F2005"> </span><span style="background-color: #C9A441"> </span><span style="background-color: #FFE05C"> </span><span style="background-color: #FEDF5B"> </span><span style="background-color: #FCDC59"> </span><span style="background-color: #F7D555"> </span><span style="background-color: #E5C04A"> </span><span style="background-color: #795E3B"> </span><span style="background-color: #726256"> </span><span style="background-color: #755F4C"> </span><span style="background-color: #A17124"> </span><span style="background-color: #AE7414"> </span><span style="background-color: #AE791D"> </span><span style="background-color: #794D18"> </span><span> </span>
|
||||||
|
<span style="background-color: #4E1F04"> </span><span style="background-color: #B78D31"> </span><span style="background-color: #DDB33E"> </span><span style="background-color: #D0A132"> </span><span style="background-color: #C28F25"> </span><span style="background-color: #B67E1A"> </span><span style="background-color: #AC7111"> </span><span style="background-color: #9E610A"> </span><span style="background-color: #5F3212"> </span><span style="background-color: #6A574B"> </span><span style="background-color: #726256"> </span><span style="background-color: #744D2A"> </span><span style="background-color: #955401"> </span><span style="background-color: #8C5106"> </span><span style="background-color: #5F310C"> </span><span> </span>
|
||||||
|
<span style="background-color: #4B1A00"> </span><span style="background-color: #854903"> </span><span style="background-color: #9B5A02"> </span><span style="background-color: #995700"> </span><span style="background-color: #935200"> </span><span style="background-color: #592402"> </span><span style="background-color: #5B3F30"> </span><span style="background-color: #726256"> </span><span style="background-color: #705A4A"> </span><span style="background-color: #844C0C"> </span><span style="background-color: #824400"> </span><span style="background-color: #4C1B00"> </span><span> </span>
|
||||||
|
<span style="background-color: #4B1A00"> </span><span style="background-color: #824500"> </span><span style="background-color: #995700"> </span><span style="background-color: #935200"> </span><span style="background-color: #592300"> </span><span style="background-color: #4F2411"> </span><span style="background-color: #6B584C"> </span><span style="background-color: #736256"> </span><span style="background-color: #734E2C"> </span><span style="background-color: #7C4101"> </span><span style="background-color: #4C1B00"> </span><span> </span>
|
||||||
|
<span style="background-color: #4B1A00"> </span><span style="background-color: #824500"> </span><span style="background-color: #995700"> </span><span style="background-color: #935200"> </span><span style="background-color: #592300"> </span><span style="background-color: #4A1902"> </span><span style="background-color: #5C4031"> </span><span style="background-color: #726256"> </span><span style="background-color: #705B4B"> </span><span style="background-color: #6A390F"> </span><span style="background-color: #4C1A00"> </span><span> </span>
|
||||||
|
<span style="background-color: #4B1A00"> </span><span style="background-color: #824500"> </span><span style="background-color: #995700"> </span><span style="background-color: #935200"> </span><span style="background-color: #592300"> </span><span style="background-color: #4A1700"> </span><span style="background-color: #4F2512"> </span><span style="background-color: #6B594D"> </span><span style="background-color: #736256"> </span><span style="background-color: #634432"> </span><span style="background-color: #4C1D08"> </span><span> </span>
|
||||||
|
<span style="background-color: #4B1A00"> </span><span style="background-color: #814400"> </span><span style="background-color: #955400"> </span><span style="background-color: #915100"> </span><span style="background-color: #8C4D00"> </span><span style="background-color: #864800"> </span><span style="background-color: #7F4301"> </span><span style="background-color: #743A01"> </span><span style="background-color: #521E01"> </span><span style="background-color: #4A1700"> </span><span style="background-color: #4A1902"> </span><span style="background-color: #5D4132"> </span><span style="background-color: #726256"> </span><span style="background-color: #6F5B4E"> </span><span style="background-color: #5D3A28"> </span><span style="background-color: #53220C"> </span>
|
||||||
|
<span style="background-color: #471801"> </span><span style="background-color: #642D01"> </span><span style="background-color: #6B3301"> </span><span style="background-color: #642E02"> </span><span style="background-color: #5D2902"> </span><span style="background-color: #542203"> </span><span style="background-color: #4C1C04"> </span><span style="background-color: #461905"> </span><span style="background-color: #4A1C07"> </span><span style="background-color: #4C1A03"> </span><span style="background-color: #4B1801"> </span><span style="background-color: #502613"> </span><span style="background-color: #69564A"> </span><span style="background-color: #705F54"> </span><span style="background-color: #604232"> </span><span style="background-color: #51200A"> </span>
|
||||||
|
<span style="background-color: #411806"> </span><span style="background-color: #431A07"> </span><span style="background-color: #411D0D"> </span><span> </span><span style="background-color: #4D1B05"> </span><span style="background-color: #4D1D07"> </span><span style="background-color: #533324"> </span><span style="background-color: #583E30"> </span><span style="background-color: #53301F"> </span><span style="background-color: #53230D"> </span>
|
||||||
|
</pre>
|
||||||
|
|
||||||
|
# Manipulating images
|
||||||
|
|
||||||
|
You can take full advantage of [ImageSharp](https://github.com/SixLabors/ImageSharp)
|
||||||
|
and manipulate images directly via it's [Processing API](https://docs.sixlabors.com/api/ImageSharp/SixLabors.ImageSharp.Processing.html).
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Load an image
|
||||||
|
var image = new CanvasImage("cake.png");
|
||||||
|
image.MaxWidth(32);
|
||||||
|
|
||||||
|
// Set a sampler that will be used when scaling the image.
|
||||||
|
image.BilinearResampler();
|
||||||
|
|
||||||
|
// Mutate the image using ImageSharp
|
||||||
|
image.Mutate(ctx => ctx.Grayscale().Rotate(-45).EntropyCrop());
|
||||||
|
|
||||||
|
// Render the image to the console
|
||||||
|
AnsiConsole.Render(image);
|
||||||
|
```
|
||||||
|
|
||||||
|
## Result
|
||||||
|
|
||||||
|
<pre style="font-size:90%;font-family:consolas,'Courier New',monospace;line-height: normal; padding: 0px;background-color: #222222; padding: 20px;">
|
||||||
|
<span> </span><span style="background-color: #282828"> </span><span style="background-color: #222222"> </span><span style="background-color: #232323"> </span><span style="background-color: #353535"> </span><span style="background-color: #4B4B4B"> </span><span style="background-color: #595959"> </span><span style="background-color: #3B3B3B"> </span><span style="background-color: #202020"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #343434"> </span><span style="background-color: #2B2B2B"> </span><span style="background-color: #292929"> </span><span style="background-color: #272727"> </span><span style="background-color: #252525"> </span><span style="background-color: #292929"> </span><span style="background-color: #555555"> </span><span style="background-color: #929292"> </span><span style="background-color: #C7C7C7"> </span><span style="background-color: #E5E5E5"> </span><span style="background-color: #F0F0F0"> </span><span style="background-color: #E4E4E4"> </span><span style="background-color: #A8A8A8"> </span><span style="background-color: #515151"> </span><span style="background-color: #202020"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #2E2E2E"> </span><span style="background-color: #2B2B2B"> </span><span style="background-color: #333333"> </span><span style="background-color: #373737"> </span><span style="background-color: #3C3C3C"> </span><span style="background-color: #414141"> </span><span style="background-color: #474747"> </span><span style="background-color: #4B4B4B"> </span><span style="background-color: #454545"> </span><span style="background-color: #828282"> </span><span style="background-color: #E0E0E0"> </span><span style="background-color: #FFFFFF"> </span><span style="background-color: #FCFCFC"> </span><span style="background-color: #DEDEDE"> </span><span style="background-color: #DADADA"> </span><span style="background-color: #BCBCBC"> </span><span style="background-color: #515151"> </span><span style="background-color: #202020"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #272727"> </span><span style="background-color: #414141"> </span><span style="background-color: #5C5C5C"> </span><span style="background-color: #616161"> </span><span style="background-color: #636363"> </span><span style="background-color: #656565"> </span><span style="background-color: #666666"> </span><span style="background-color: #656565"> </span><span style="background-color: #5A5A5A"> </span><span style="background-color: #707070"> </span><span style="background-color: #F3F3F3"> </span><span style="background-color: #FFFFFF"> </span><span style="background-color: #F0F0F0"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #BABABA"> </span><span style="background-color: #505050"> </span><span style="background-color: #202020"> </span><span style="background-color: #1B1B1B"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #242424"> </span><span style="background-color: #3B3B3B"> </span><span style="background-color: #545454"> </span><span style="background-color: #606060"> </span><span style="background-color: #656565"> </span><span style="background-color: #666666"> </span><span style="background-color: #606060"> </span><span style="background-color: #575757"> </span><span style="background-color: #E8E8E8"> </span><span style="background-color: #F6F6F6"> </span><span style="background-color: #E1E1E1"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #D9D9D9"> </span><span style="background-color: #A0A0A0"> </span><span style="background-color: #989898"> </span><span style="background-color: #4E4E4E"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
<span style="background-color: #2F2F2F"> </span><span style="background-color: #2C2C2C"> </span><span style="background-color: #222222"> </span><span style="background-color: #282828"> </span><span style="background-color: #2D2D2D"> </span><span style="background-color: #3E3E3E"> </span><span style="background-color: #4D4D4D"> </span><span style="background-color: #616161"> </span><span style="background-color: #636363"> </span><span style="background-color: #666666"> </span><span style="background-color: #606060"> </span><span style="background-color: #535353"> </span><span style="background-color: #D4D4D4"> </span><span style="background-color: #E2E2E2"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #DCDCDC"> </span><span style="background-color: #AFAFAF"> </span><span style="background-color: #666666"> </span><span style="background-color: #6F6F6F"> </span><span style="background-color: #717171"> </span><span style="background-color: #242424"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span style="background-color: #2C2C2C"> </span><span style="background-color: #343434"> </span><span style="background-color: #2E2E2E"> </span><span style="background-color: #262626"> </span><span style="background-color: #404040"> </span><span style="background-color: #868686"> </span><span style="background-color: #4D4D4D"> </span><span style="background-color: #5A5A5A"> </span><span style="background-color: #3D3D3D"> </span><span style="background-color: #474747"> </span><span style="background-color: #646464"> </span><span style="background-color: #616161"> </span><span style="background-color: #4D4D4D"> </span><span style="background-color: #9D9D9D"> </span><span style="background-color: #C8C8C8"> </span><span style="background-color: #DADADA"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #C4C4C4"> </span><span style="background-color: #717171"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #595959"> </span><span style="background-color: #343434"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span style="background-color: #343434"> </span><span style="background-color: #575757"> </span><span style="background-color: #555555"> </span><span style="background-color: #454545"> </span><span style="background-color: #4C4C4C"> </span><span style="background-color: #656565"> </span><span style="background-color: #5B5B5B"> </span><span style="background-color: #434343"> </span><span style="background-color: #3E3E3E"> </span><span style="background-color: #595959"> </span><span style="background-color: #666666"> </span><span style="background-color: #606060"> </span><span style="background-color: #595959"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #787878"> </span><span style="background-color: #9E9E9E"> </span><span style="background-color: #797979"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #575757"> </span><span style="background-color: #343434"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span style="background-color: #2B2B2B"> </span><span style="background-color: #3B3B3B"> </span><span style="background-color: #575757"> </span><span style="background-color: #646464"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #575757"> </span><span style="background-color: #3D3D3D"> </span><span style="background-color: #525252"> </span><span style="background-color: #656565"> </span><span style="background-color: #666666"> </span><span style="background-color: #656565"> </span><span style="background-color: #616161"> </span><span style="background-color: #595959"> </span><span style="background-color: #4B4B4B"> </span><span style="background-color: #454545"> </span><span style="background-color: #4B4B4B"> </span><span style="background-color: #555555"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #575757"> </span><span style="background-color: #343434"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span style="background-color: #3A3A3A"> </span><span style="background-color: #292929"> </span><span style="background-color: #323232"> </span><span style="background-color: #4A4A4A"> </span><span style="background-color: #626262"> </span><span style="background-color: #666666"> </span><span style="background-color: #656565"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #616161"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #616161"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #666666"> </span><span style="background-color: #626262"> </span><span style="background-color: #575757"> </span><span style="background-color: #4B4B4B"> </span><span style="background-color: #454545"> </span><span style="background-color: #4A4A4A"> </span><span style="background-color: #545454"> </span><span style="background-color: #343434"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #191919"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #252525"> </span><span style="background-color: #383838"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #616161"> </span><span style="background-color: #5B5B5B"> </span><span style="background-color: #505050"> </span><span style="background-color: #545454"> </span><span style="background-color: #8A8A8A"> </span><span style="background-color: #C5C5C5"> </span><span style="background-color: #959595"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #636363"> </span><span style="background-color: #666666"> </span><span style="background-color: #626262"> </span><span style="background-color: #595959"> </span><span style="background-color: #4D4D4D"> </span><span style="background-color: #454545"> </span><span style="background-color: #414141"> </span><span style="background-color: #282828"> </span><span style="background-color: #1E1E1E"> </span><span style="background-color: #1D1D1D"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #212121"> </span><span style="background-color: #2C2C2C"> </span><span style="background-color: #4F4F4F"> </span><span style="background-color: #515151"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #898989"> </span><span style="background-color: #CDCDCD"> </span><span style="background-color: #E8E8E8"> </span><span style="background-color: #DEDEDE"> </span><span style="background-color: #D8D8D8"> </span><span style="background-color: #939393"> </span><span style="background-color: #4D4D4D"> </span><span style="background-color: #525252"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #646464"> </span><span style="background-color: #666666"> </span><span style="background-color: #636363"> </span><span style="background-color: #5A5A5A"> </span><span style="background-color: #4A4A4A"> </span><span style="background-color: #383838"> </span><span style="background-color: #323232"> </span><span style="background-color: #2A2A2A"> </span><span style="background-color: #282828"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #272727"> </span><span style="background-color: #404040"> </span><span style="background-color: #C8C8C8"> </span><span style="background-color: #DFDFDF"> </span><span style="background-color: #F0F0F0"> </span><span style="background-color: #FDFDFD"> </span><span style="background-color: #F3F3F3"> </span><span style="background-color: #DFDFDF"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #D7D7D7"> </span><span style="background-color: #757575"> </span><span style="background-color: #2B2B2B"> </span><span style="background-color: #333333"> </span><span style="background-color: #444444"> </span><span style="background-color: #535353"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #646464"> </span><span style="background-color: #666666"> </span><span style="background-color: #646464"> </span><span style="background-color: #5B5B5B"> </span><span style="background-color: #4F4F4F"> </span><span style="background-color: #3A3A3A"> </span><span style="background-color: #292929"> </span>
|
||||||
|
<span> </span><span style="background-color: #242424"> </span><span style="background-color: #4F4F4F"> </span><span style="background-color: #E7E7E7"> </span><span style="background-color: #FFFFFF"> </span><span style="background-color: #F2F2F2"> </span><span style="background-color: #DFDFDF"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #C2C2C2"> </span><span style="background-color: #6E6E6E"> </span><span style="background-color: #434343"> </span><span style="background-color: #242424"> </span><span style="background-color: #222222"> </span><span style="background-color: #282828"> </span><span style="background-color: #343434"> </span><span style="background-color: #454545"> </span><span style="background-color: #555555"> </span><span style="background-color: #606060"> </span><span style="background-color: #656565"> </span><span style="background-color: #666666"> </span><span style="background-color: #595959"> </span><span style="background-color: #313131"> </span>
|
||||||
|
<span> </span><span style="background-color: #222222"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #F2F2F2"> </span><span style="background-color: #FFFFFF"> </span><span style="background-color: #F4F4F4"> </span><span style="background-color: #D7D7D7"> </span><span style="background-color: #DCDCDC"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #D1D1D1"> </span><span style="background-color: #818181"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #434343"> </span><span style="background-color: #242424"> </span><span style="background-color: #202020"> </span><span style="background-color: #222222"> </span><span style="background-color: #282828"> </span><span style="background-color: #353535"> </span><span style="background-color: #464646"> </span><span style="background-color: #565656"> </span><span style="background-color: #606060"> </span><span style="background-color: #656565"> </span><span style="background-color: #666666"> </span><span style="background-color: #585858"> </span><span style="background-color: #333333"> </span>
|
||||||
|
<span> </span><span style="background-color: #222222"> </span><span style="background-color: #707070"> </span><span style="background-color: #FAFAFA"> </span><span style="background-color: #D2D2D2"> </span><span style="background-color: #D9D9D9"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #D9D9D9"> </span><span style="background-color: #979797"> </span><span style="background-color: #616161"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #434343"> </span><span style="background-color: #242424"> </span><span style="background-color: #202020"> </span><span style="background-color: #222222"> </span><span style="background-color: #292929"> </span><span style="background-color: #363636"> </span><span style="background-color: #474747"> </span><span style="background-color: #575757"> </span><span style="background-color: #606060"> </span><span style="background-color: #616161"> </span><span style="background-color: #575757"> </span><span style="background-color: #404040"> </span><span style="background-color: #2B2B2B"> </span>
|
||||||
|
<span> </span><span style="background-color: #212121"> </span><span style="background-color: #858585"> </span><span style="background-color: #FCFCFC"> </span><span style="background-color: #D9D9D9"> </span><span style="background-color: #D2D2D2"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #DCDCDC"> </span><span style="background-color: #AEAEAE"> </span><span style="background-color: #666666"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #434343"> </span><span style="background-color: #242424"> </span><span style="background-color: #202020"> </span><span style="background-color: #222222"> </span><span style="background-color: #292929"> </span><span style="background-color: #363636"> </span><span style="background-color: #3E3E3E"> </span><span style="background-color: #363636"> </span><span style="background-color: #2B2B2B"> </span><span style="background-color: #282828"> </span>
|
||||||
|
<span> </span><span style="background-color: #222222"> </span><span style="background-color: #9B9B9B"> </span><span style="background-color: #EAEAEA"> </span><span style="background-color: #D0D0D0"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #C3C3C3"> </span><span style="background-color: #707070"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #434343"> </span><span style="background-color: #242424"> </span><span style="background-color: #202020"> </span><span style="background-color: #212121"> </span><span style="background-color: #242424"> </span><span style="background-color: #272727"> </span><span style="background-color: #2C2C2C"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #292929"> </span><span style="background-color: #ACACAC"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #DCDCDC"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #D1D1D1"> </span><span style="background-color: #818181"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #434343"> </span><span style="background-color: #242424"> </span><span style="background-color: #202020"> </span><span style="background-color: #212121"> </span><span style="background-color: #222222"> </span><span style="background-color: #232323"> </span><span style="background-color: #242424"> </span><span style="background-color: #262626"> </span><span style="background-color: #2E2E2E"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #2D2D2D"> </span><span style="background-color: #A6A6A6"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #D9D9D9"> </span><span style="background-color: #989898"> </span><span style="background-color: #616161"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5D5D5D"> </span><span style="background-color: #3E3E3E"> </span><span style="background-color: #222222"> </span><span style="background-color: #242424"> </span><span style="background-color: #262626"> </span><span style="background-color: #2B2B2B"> </span><span style="background-color: #363636"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #212121"> </span><span style="background-color: #575757"> </span><span style="background-color: #BEBEBE"> </span><span style="background-color: #DDDDDD"> </span><span style="background-color: #DCDCDC"> </span><span style="background-color: #AFAFAF"> </span><span style="background-color: #666666"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5B5B5B"> </span><span style="background-color: #373737"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #212121"> </span><span style="background-color: #585858"> </span><span style="background-color: #BEBEBE"> </span><span style="background-color: #C3C3C3"> </span><span style="background-color: #717171"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #424242"> </span><span style="background-color: #252525"> </span><span style="background-color: #242424"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #212121"> </span><span style="background-color: #545454"> </span><span style="background-color: #717171"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #4D4D4D"> </span><span style="background-color: #292929"> </span><span style="background-color: #232323"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #565656"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #565656"> </span><span style="background-color: #303030"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #565656"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5C5C5C"> </span><span style="background-color: #393939"> </span><span style="background-color: #232323"> </span><span style="background-color: #252525"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #565656"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #5E5E5E"> </span><span style="background-color: #444444"> </span><span style="background-color: #252525"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #565656"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #4F4F4F"> </span><span style="background-color: #2A2A2A"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #565656"> </span><span style="background-color: #5F5F5F"> </span><span style="background-color: #575757"> </span><span style="background-color: #323232"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #565656"> </span><span style="background-color: #5C5C5C"> </span><span style="background-color: #3C3C3C"> </span><span style="background-color: #232323"> </span><span style="background-color: #252525"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1F1F1F"> </span><span style="background-color: #343434"> </span><span style="background-color: #404040"> </span><span style="background-color: #262626"> </span><span style="background-color: #232323"> </span><span> </span>
|
||||||
|
<span> </span><span style="background-color: #171717"> </span><span style="background-color: #1E1E1E"> </span><span style="background-color: #222222"> </span><span> </span>
|
||||||
|
</pre>
|
||||||
52
docs/input/widgets/canvas.md
Normal file
52
docs/input/widgets/canvas.md
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
Title: Canvas
|
||||||
|
Order: 4
|
||||||
|
---
|
||||||
|
|
||||||
|
`Canvas` is a widget that allows you to render arbitrary "pixels"
|
||||||
|
(or _coxels_, as [Simon Cropp](https://twitter.com/SimonCropp/status/1331554791726534657?s=20)
|
||||||
|
suggested we should call them).
|
||||||
|
|
||||||
|
# Drawing primitives
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Create a canvas
|
||||||
|
var canvas = new Canvas(16, 16);
|
||||||
|
|
||||||
|
// Draw some shapes
|
||||||
|
for(var i = 0; i < canvas.Width; i++)
|
||||||
|
{
|
||||||
|
// Cross
|
||||||
|
canvas.SetPixel(i, i, Color.White);
|
||||||
|
canvas.SetPixel(canvas.Width - i - 1, i, Color.White);
|
||||||
|
|
||||||
|
// Border
|
||||||
|
canvas.SetPixel(i, 0, Color.Red);
|
||||||
|
canvas.SetPixel(0, i, Color.Green);
|
||||||
|
canvas.SetPixel(i, canvas.Height - 1, Color.Blue);
|
||||||
|
canvas.SetPixel(canvas.Width - 1, i, Color.Yellow);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Render the canvas
|
||||||
|
AnsiConsole.Render(canvas);
|
||||||
|
```
|
||||||
|
|
||||||
|
## Result
|
||||||
|
|
||||||
|
<pre style="font-size:100%;font-family:consolas,'Courier New',monospace;line-height: normal; padding: 0px;background-color: #222222; padding: 20px;">
|
||||||
|
<span style="background-color: #008000"> </span><span style="background-color: #FF0000"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span style="background-color: #800080"> </span><span> </span><span style="background-color: #800080"> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
<span style="background-color: #008000"> </span><span style="background-color: #0000FF"> </span><span style="background-color: #FFFF00"> </span>
|
||||||
|
</pre>
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
Title: Figlet
|
Title: Figlet
|
||||||
Order: 5
|
Order: 3
|
||||||
|
RedirectFrom: figlet
|
||||||
---
|
---
|
||||||
|
|
||||||
Spectre.Console can render [FIGlet](http://www.figlet.org/) text by using the `FigletText` class.
|
Spectre.Console can render [FIGlet](http://www.figlet.org/) text by using the `FigletText` class.
|
||||||
12
docs/input/widgets/index.cshtml
Normal file
12
docs/input/widgets/index.cshtml
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
Title: Widgets
|
||||||
|
Order: 9
|
||||||
|
---
|
||||||
|
|
||||||
|
<h1>Sections</h1>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
@foreach (IDocument child in OutputPages.GetChildrenOf(Document))
|
||||||
|
{
|
||||||
|
<li>@Html.DocumentLink(child)</li>
|
||||||
|
}
|
||||||
|
</ul>
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
Title: Widgets
|
|
||||||
Order: 9
|
|
||||||
---
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
Title: Rule
|
Title: Rule
|
||||||
Order: 5
|
Order: 1
|
||||||
RedirectFrom: rule
|
RedirectFrom: rule
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
Title: Table
|
Title: Table
|
||||||
Order: 3
|
Order: 0
|
||||||
RedirectFrom: tables
|
RedirectFrom: tables
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
22
examples/Canvas/Canvas.csproj
Normal file
22
examples/Canvas/Canvas.csproj
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||||
|
<IsPackable>false</IsPackable>
|
||||||
|
<Title>Canvas</Title>
|
||||||
|
<Description>Demonstrates how to render pixels and images.</Description>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\..\src\Spectre.Console.ImageSharp\Spectre.Console.ImageSharp.csproj" />
|
||||||
|
<ProjectReference Include="..\..\src\Spectre.Console\Spectre.Console.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<None Update="cake.png">
|
||||||
|
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
87
examples/Canvas/Mandelbrot.cs
Normal file
87
examples/Canvas/Mandelbrot.cs
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
/*
|
||||||
|
Ported from: https://rosettacode.org/wiki/Mandelbrot_set#C.23
|
||||||
|
Licensed under GNU Free Documentation License 1.2
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using Spectre.Console;
|
||||||
|
|
||||||
|
namespace CanvasExample
|
||||||
|
{
|
||||||
|
public static class Mandelbrot
|
||||||
|
{
|
||||||
|
private const double MaxValueExtent = 2.0;
|
||||||
|
|
||||||
|
private struct ComplexNumber
|
||||||
|
{
|
||||||
|
public double Real { get; }
|
||||||
|
public double Imaginary { get; }
|
||||||
|
|
||||||
|
public ComplexNumber(double real, double imaginary)
|
||||||
|
{
|
||||||
|
Real = real;
|
||||||
|
Imaginary = imaginary;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ComplexNumber operator +(ComplexNumber x, ComplexNumber y)
|
||||||
|
{
|
||||||
|
return new ComplexNumber(x.Real + y.Real, x.Imaginary + y.Imaginary);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ComplexNumber operator *(ComplexNumber x, ComplexNumber y)
|
||||||
|
{
|
||||||
|
return new ComplexNumber(x.Real * y.Real - x.Imaginary * y.Imaginary,
|
||||||
|
x.Real * y.Imaginary + x.Imaginary * y.Real);
|
||||||
|
}
|
||||||
|
|
||||||
|
public double Abs()
|
||||||
|
{
|
||||||
|
return Real * Real + Imaginary * Imaginary;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Canvas Generate(int width, int height)
|
||||||
|
{
|
||||||
|
var canvas = new Canvas(width, height);
|
||||||
|
|
||||||
|
var scale = 2 * MaxValueExtent / Math.Min(canvas.Width, canvas.Height);
|
||||||
|
for (var i = 0; i < canvas.Height; i++)
|
||||||
|
{
|
||||||
|
var y = (canvas.Height / 2 - i) * scale;
|
||||||
|
for (var j = 0; j < canvas.Width; j++)
|
||||||
|
{
|
||||||
|
var x = (j - canvas.Width / 2) * scale;
|
||||||
|
var value = Calculate(new ComplexNumber(x, y));
|
||||||
|
canvas.SetPixel(j, i, GetColor(value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return canvas;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static double Calculate(ComplexNumber c)
|
||||||
|
{
|
||||||
|
const int MaxIterations = 1000;
|
||||||
|
const double MaxNorm = MaxValueExtent * MaxValueExtent;
|
||||||
|
|
||||||
|
var iteration = 0;
|
||||||
|
var z = new ComplexNumber();
|
||||||
|
do
|
||||||
|
{
|
||||||
|
z = z * z + c;
|
||||||
|
iteration++;
|
||||||
|
} while (z.Abs() < MaxNorm && iteration < MaxIterations);
|
||||||
|
|
||||||
|
return iteration < MaxIterations
|
||||||
|
? (double)iteration / MaxIterations
|
||||||
|
: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Color GetColor(double value)
|
||||||
|
{
|
||||||
|
const double MaxColor = 256;
|
||||||
|
const double ContrastValue = 0.2;
|
||||||
|
return new Color(0, 0, (byte)(MaxColor * Math.Pow(value, ContrastValue)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
36
examples/Canvas/Program.cs
Normal file
36
examples/Canvas/Program.cs
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
using SixLabors.ImageSharp.Processing;
|
||||||
|
using Spectre.Console;
|
||||||
|
using Spectre.Console.Rendering;
|
||||||
|
|
||||||
|
namespace CanvasExample
|
||||||
|
{
|
||||||
|
public static class Program
|
||||||
|
{
|
||||||
|
public static void Main()
|
||||||
|
{
|
||||||
|
// Draw a mandelbrot set using a Canvas
|
||||||
|
var mandelbrot = Mandelbrot.Generate(32, 32);
|
||||||
|
Render(mandelbrot, "Mandelbrot");
|
||||||
|
|
||||||
|
// Draw an image using CanvasImage powered by ImageSharp.
|
||||||
|
// This requires the "Spectre.Console.ImageSharp" NuGet package.
|
||||||
|
var image = new CanvasImage("cake.png");
|
||||||
|
image.BilinearResampler();
|
||||||
|
image.MaxWidth(16);
|
||||||
|
Render(image, "Image from file (16 wide)");
|
||||||
|
|
||||||
|
// Draw image again, but without render width
|
||||||
|
image.NoMaxWidth();
|
||||||
|
image.Mutate(ctx => ctx.Grayscale().Rotate(-45).EntropyCrop());
|
||||||
|
Render(image, "Image from file (fit, greyscale, rotated)");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Render(IRenderable canvas, string title)
|
||||||
|
{
|
||||||
|
AnsiConsole.WriteLine();
|
||||||
|
AnsiConsole.Render(new Rule($"[yellow]{title}[/]").LeftAligned().RuleStyle("grey"));
|
||||||
|
AnsiConsole.WriteLine();
|
||||||
|
AnsiConsole.Render(canvas);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
BIN
examples/Canvas/cake.png
Normal file
BIN
examples/Canvas/cake.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 52 KiB |
@@ -1,39 +1,31 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Net.Http;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Newtonsoft.Json.Linq;
|
|
||||||
using Spectre.Console;
|
using Spectre.Console;
|
||||||
|
|
||||||
namespace ColumnsExample
|
namespace ColumnsExample
|
||||||
{
|
{
|
||||||
public static class Program
|
public static class Program
|
||||||
{
|
{
|
||||||
public static async Task Main()
|
public static void Main()
|
||||||
{
|
{
|
||||||
// Download some random users
|
|
||||||
using var client = new HttpClient();
|
|
||||||
dynamic users = JObject.Parse(
|
|
||||||
await client.GetStringAsync("https://randomuser.me/api/?results=15"));
|
|
||||||
|
|
||||||
// Create a card for each user
|
|
||||||
var cards = new List<Panel>();
|
var cards = new List<Panel>();
|
||||||
foreach(var user in users.results)
|
foreach(var user in User.LoadUsers())
|
||||||
{
|
{
|
||||||
cards.Add(new Panel(GetCardContent(user))
|
cards.Add(
|
||||||
.Header($"{user.location.country}")
|
new Panel(GetCardContent(user))
|
||||||
.RoundedBorder().Expand());
|
.Header($"{user.Country}")
|
||||||
|
.RoundedBorder().Expand());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Render all cards in columns
|
// Render all cards in columns
|
||||||
AnsiConsole.Render(new Columns(cards));
|
AnsiConsole.Render(new Columns(cards));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static string GetCardContent(dynamic user)
|
private static string GetCardContent(User user)
|
||||||
{
|
{
|
||||||
var name = $"{user.name.first} {user.name.last}";
|
var name = $"{user.FirstName} {user.LastName}";
|
||||||
var country = $"{user.location.city}";
|
var city = $"{user.City}";
|
||||||
|
|
||||||
return $"[b]{name}[/]\n[yellow]{country}[/]";
|
return $"[b]{name}[/]\n[yellow]{city}[/]";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
89
examples/Columns/User.cs
Normal file
89
examples/Columns/User.cs
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace ColumnsExample
|
||||||
|
{
|
||||||
|
public sealed class User
|
||||||
|
{
|
||||||
|
public string FirstName { get; set; }
|
||||||
|
public string LastName { get; set; }
|
||||||
|
public string City { get; set; }
|
||||||
|
public string Country { get; set; }
|
||||||
|
|
||||||
|
public static List<User> LoadUsers()
|
||||||
|
{
|
||||||
|
return new List<User>
|
||||||
|
{
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Andrea",
|
||||||
|
LastName = "Johansen",
|
||||||
|
City = "Hornbæk",
|
||||||
|
Country = "Denmark",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Brandon",
|
||||||
|
LastName = "Cole",
|
||||||
|
City = "Washington",
|
||||||
|
Country = "United States",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Patrik",
|
||||||
|
LastName = "Svensson",
|
||||||
|
City = "Stockholm",
|
||||||
|
Country = "Sweden",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Freya",
|
||||||
|
LastName = "Thompson",
|
||||||
|
City = "Rotorua",
|
||||||
|
Country = "New Zealand",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "طاها",
|
||||||
|
LastName = "رضایی",
|
||||||
|
City = "اهواز",
|
||||||
|
Country = "Iran",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Yara",
|
||||||
|
LastName = "Simon",
|
||||||
|
City = "Develier",
|
||||||
|
Country = "Switzerland",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Giray",
|
||||||
|
LastName = "Erbay",
|
||||||
|
City = "Karabük",
|
||||||
|
Country = "Turkey",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Miodrag",
|
||||||
|
LastName = "Schaffer",
|
||||||
|
City = "Möckern",
|
||||||
|
Country = "Germany",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Carmela",
|
||||||
|
LastName = "Lo Castro",
|
||||||
|
City = "Firenze",
|
||||||
|
Country = "Italy",
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
FirstName = "Roberto",
|
||||||
|
LastName = "Sims",
|
||||||
|
City = "Mallow",
|
||||||
|
Country = "Ireland",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
using System;
|
|
||||||
using Spectre.Console;
|
using Spectre.Console;
|
||||||
|
|
||||||
namespace InfoExample
|
namespace InfoExample
|
||||||
@@ -13,7 +12,7 @@ namespace InfoExample
|
|||||||
.AddRow("[b]Color system[/]", $"{AnsiConsole.Capabilities.ColorSystem}")
|
.AddRow("[b]Color system[/]", $"{AnsiConsole.Capabilities.ColorSystem}")
|
||||||
.AddRow("[b]Supports ansi?[/]", $"{YesNo(AnsiConsole.Capabilities.SupportsAnsi)}")
|
.AddRow("[b]Supports ansi?[/]", $"{YesNo(AnsiConsole.Capabilities.SupportsAnsi)}")
|
||||||
.AddRow("[b]Legacy console?[/]", $"{YesNo(AnsiConsole.Capabilities.LegacyConsole)}")
|
.AddRow("[b]Legacy console?[/]", $"{YesNo(AnsiConsole.Capabilities.LegacyConsole)}")
|
||||||
.AddRow("[b]Interactive?[/]", $"{YesNo(Environment.UserInteractive)}")
|
.AddRow("[b]Interactive?[/]", $"{YesNo(AnsiConsole.Capabilities.SupportsInteraction)}")
|
||||||
.AddRow("[b]Buffer width[/]", $"{AnsiConsole.Console.Width}")
|
.AddRow("[b]Buffer width[/]", $"{AnsiConsole.Console.Width}")
|
||||||
.AddRow("[b]Buffer height[/]", $"{AnsiConsole.Console.Height}");
|
.AddRow("[b]Buffer height[/]", $"{AnsiConsole.Console.Height}");
|
||||||
|
|
||||||
|
|||||||
45
examples/Progress/DescriptionGenerator.cs
Normal file
45
examples/Progress/DescriptionGenerator.cs
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace ProgressExample
|
||||||
|
{
|
||||||
|
public static class DescriptionGenerator
|
||||||
|
{
|
||||||
|
private static readonly string[] _verbs = new[] { "Downloading", "Rerouting", "Retriculating", "Collapsing", "Folding", "Solving", "Colliding", "Measuring" };
|
||||||
|
private static readonly string[] _nouns = new[] { "internet", "splines", "space", "capacitators", "quarks", "algorithms", "data structures", "spacetime" };
|
||||||
|
|
||||||
|
private static readonly Random _random;
|
||||||
|
private static readonly HashSet<string> _used;
|
||||||
|
|
||||||
|
static DescriptionGenerator()
|
||||||
|
{
|
||||||
|
_random = new Random(DateTime.Now.Millisecond);
|
||||||
|
_used = new HashSet<string>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool TryGenerate(out string name)
|
||||||
|
{
|
||||||
|
var iterations = 0;
|
||||||
|
while (iterations < 25)
|
||||||
|
{
|
||||||
|
name = Generate();
|
||||||
|
if (!_used.Contains(name))
|
||||||
|
{
|
||||||
|
_used.Add(name);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
iterations++;
|
||||||
|
}
|
||||||
|
|
||||||
|
name = Generate();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string Generate()
|
||||||
|
{
|
||||||
|
return _verbs[_random.Next(0, _verbs.Length)]
|
||||||
|
+ " " + _nouns[_random.Next(0, _nouns.Length)];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
75
examples/Progress/Program.cs
Normal file
75
examples/Progress/Program.cs
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading;
|
||||||
|
using Spectre.Console;
|
||||||
|
|
||||||
|
namespace ProgressExample
|
||||||
|
{
|
||||||
|
public static class Program
|
||||||
|
{
|
||||||
|
public static void Main()
|
||||||
|
{
|
||||||
|
AnsiConsole.MarkupLine("[yellow]Initializing warp drive[/]...");
|
||||||
|
|
||||||
|
// Show progress
|
||||||
|
AnsiConsole.Progress()
|
||||||
|
.AutoClear(false)
|
||||||
|
.Columns(new ProgressColumn[]
|
||||||
|
{
|
||||||
|
new TaskDescriptionColumn(), // Task description
|
||||||
|
new ProgressBarColumn(), // Progress bar
|
||||||
|
new PercentageColumn(), // Percentage
|
||||||
|
new RemainingTimeColumn(), // Remaining time
|
||||||
|
new SpinnerColumn(), // Spinner
|
||||||
|
})
|
||||||
|
.Start(ctx =>
|
||||||
|
{
|
||||||
|
var random = new Random(DateTime.Now.Millisecond);
|
||||||
|
var tasks = CreateTasks(ctx, random);
|
||||||
|
|
||||||
|
while (!ctx.IsFinished)
|
||||||
|
{
|
||||||
|
// Increment progress
|
||||||
|
foreach (var (task, increment) in tasks)
|
||||||
|
{
|
||||||
|
task.Increment(random.NextDouble() * increment);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write some random things to the terminal
|
||||||
|
if (random.NextDouble() < 0.1)
|
||||||
|
{
|
||||||
|
WriteLogMessage();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Simulate some delay
|
||||||
|
Thread.Sleep(100);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Done
|
||||||
|
AnsiConsole.MarkupLine("[green]Done![/]");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static List<(ProgressTask, int)> CreateTasks(ProgressContext progress, Random random)
|
||||||
|
{
|
||||||
|
var tasks = new List<(ProgressTask, int)>();
|
||||||
|
while (tasks.Count < 5)
|
||||||
|
{
|
||||||
|
if (DescriptionGenerator.TryGenerate(out var name))
|
||||||
|
{
|
||||||
|
tasks.Add((progress.AddTask(name), random.Next(2, 10)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tasks;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void WriteLogMessage()
|
||||||
|
{
|
||||||
|
AnsiConsole.MarkupLine(
|
||||||
|
"[grey]LOG:[/] " +
|
||||||
|
DescriptionGenerator.Generate() +
|
||||||
|
"[grey]...[/]");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
19
examples/Progress/Progress.csproj
Normal file
19
examples/Progress/Progress.csproj
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>net5.0</TargetFramework>
|
||||||
|
<IsPackable>false</IsPackable>
|
||||||
|
<Title>Progress</Title>
|
||||||
|
<Description>Demonstrates how to show progress bars.</Description>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Newtonsoft.Json" Version="12.0.3" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\..\src\Spectre.Console\Spectre.Console.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -6,6 +6,13 @@ namespace Cursor
|
|||||||
{
|
{
|
||||||
public static void Main(string[] args)
|
public static void Main(string[] args)
|
||||||
{
|
{
|
||||||
|
// Check if we can accept key strokes
|
||||||
|
if (!AnsiConsole.Capabilities.SupportsInteraction)
|
||||||
|
{
|
||||||
|
AnsiConsole.MarkupLine("[red]Environment does not support interaction.[/]");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Confirmation
|
// Confirmation
|
||||||
if (!AnsiConsole.Confirm("Run prompt example?"))
|
if (!AnsiConsole.Confirm("Run prompt example?"))
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -7,34 +7,34 @@ namespace EmojiExample
|
|||||||
public static void Main(string[] args)
|
public static void Main(string[] args)
|
||||||
{
|
{
|
||||||
// No title
|
// No title
|
||||||
WrapInPanel(
|
Render(
|
||||||
new Rule()
|
new Rule()
|
||||||
.RuleStyle(Style.Parse("yellow"))
|
.RuleStyle(Style.Parse("yellow"))
|
||||||
.AsciiBorder()
|
.AsciiBorder()
|
||||||
.LeftAligned());
|
.LeftAligned());
|
||||||
|
|
||||||
// Left aligned title
|
// Left aligned title
|
||||||
WrapInPanel(
|
Render(
|
||||||
new Rule("[blue]Left aligned[/]")
|
new Rule("[blue]Left aligned[/]")
|
||||||
.RuleStyle(Style.Parse("red"))
|
.RuleStyle(Style.Parse("red"))
|
||||||
.DoubleBorder()
|
.DoubleBorder()
|
||||||
.LeftAligned());
|
.LeftAligned());
|
||||||
|
|
||||||
// Centered title
|
// Centered title
|
||||||
WrapInPanel(
|
Render(
|
||||||
new Rule("[green]Centered[/]")
|
new Rule("[green]Centered[/]")
|
||||||
.RuleStyle(Style.Parse("green"))
|
.RuleStyle(Style.Parse("green"))
|
||||||
.HeavyBorder()
|
.HeavyBorder()
|
||||||
.Centered());
|
.Centered());
|
||||||
|
|
||||||
// Right aligned title
|
// Right aligned title
|
||||||
WrapInPanel(
|
Render(
|
||||||
new Rule("[red]Right aligned[/]")
|
new Rule("[red]Right aligned[/]")
|
||||||
.RuleStyle(Style.Parse("blue"))
|
.RuleStyle(Style.Parse("blue"))
|
||||||
.RightAligned());
|
.RightAligned());
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void WrapInPanel(Rule rule)
|
private static void Render(Rule rule)
|
||||||
{
|
{
|
||||||
AnsiConsole.Render(rule);
|
AnsiConsole.Render(rule);
|
||||||
AnsiConsole.WriteLine();
|
AnsiConsole.WriteLine();
|
||||||
|
|||||||
70
examples/Status/Program.cs
Normal file
70
examples/Status/Program.cs
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using Spectre.Console;
|
||||||
|
|
||||||
|
namespace ProgressExample
|
||||||
|
{
|
||||||
|
public static class Program
|
||||||
|
{
|
||||||
|
public static void Main()
|
||||||
|
{
|
||||||
|
AnsiConsole.Status()
|
||||||
|
.AutoRefresh(true)
|
||||||
|
.Spinner(Spinner.Known.Default)
|
||||||
|
.Start("[yellow]Initializing warp drive[/]", ctx =>
|
||||||
|
{
|
||||||
|
// Initialize
|
||||||
|
Thread.Sleep(3000);
|
||||||
|
WriteLogMessage("Starting gravimetric field displacement manifold");
|
||||||
|
Thread.Sleep(1000);
|
||||||
|
WriteLogMessage("Warming up deuterium chamber");
|
||||||
|
Thread.Sleep(2000);
|
||||||
|
WriteLogMessage("Generating antideuterium");
|
||||||
|
|
||||||
|
// Warp nacelles
|
||||||
|
Thread.Sleep(3000);
|
||||||
|
ctx.Spinner(Spinner.Known.BouncingBar);
|
||||||
|
ctx.Status("[bold blue]Unfolding warp nacelles[/]");
|
||||||
|
WriteLogMessage("Unfolding left warp nacelle");
|
||||||
|
Thread.Sleep(2000);
|
||||||
|
WriteLogMessage("Left warp nacelle [green]online[/]");
|
||||||
|
WriteLogMessage("Unfolding right warp nacelle");
|
||||||
|
Thread.Sleep(1000);
|
||||||
|
WriteLogMessage("Right warp nacelle [green]online[/]");
|
||||||
|
|
||||||
|
// Warp bubble
|
||||||
|
Thread.Sleep(3000);
|
||||||
|
ctx.Spinner(Spinner.Known.Star2);
|
||||||
|
ctx.Status("[bold blue]Generating warp bubble[/]");
|
||||||
|
Thread.Sleep(3000);
|
||||||
|
ctx.Spinner(Spinner.Known.Star);
|
||||||
|
ctx.Status("[bold blue]Stabilizing warp bubble[/]");
|
||||||
|
|
||||||
|
// Safety
|
||||||
|
ctx.Spinner(Spinner.Known.Monkey);
|
||||||
|
ctx.Status("[bold blue]Performing safety checks[/]");
|
||||||
|
WriteLogMessage("Enabling interior dampening");
|
||||||
|
Thread.Sleep(2000);
|
||||||
|
WriteLogMessage("Interior dampening [green]enabled[/]");
|
||||||
|
|
||||||
|
// Warp!
|
||||||
|
Thread.Sleep(3000);
|
||||||
|
ctx.Spinner(Spinner.Known.Moon);
|
||||||
|
WriteLogMessage("Preparing for warp");
|
||||||
|
Thread.Sleep(1000);
|
||||||
|
for (var warp = 1; warp < 10; warp++)
|
||||||
|
{
|
||||||
|
ctx.Status($"[bold blue]Warp {warp}[/]");
|
||||||
|
Thread.Sleep(500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Done
|
||||||
|
AnsiConsole.MarkupLine("[bold green]Crusing at Warp 9.8[/]");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void WriteLogMessage(string message)
|
||||||
|
{
|
||||||
|
AnsiConsole.MarkupLine($"[grey]LOG:[/] {message}[grey]...[/]");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
19
examples/Status/Status.csproj
Normal file
19
examples/Status/Status.csproj
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>net5.0</TargetFramework>
|
||||||
|
<IsPackable>false</IsPackable>
|
||||||
|
<Title>Status</Title>
|
||||||
|
<Description>Demonstrates how to show status updates.</Description>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Newtonsoft.Json" Version="12.0.3" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\..\src\Spectre.Console\Spectre.Console.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
22
resources/scripts/Generate-Spinners.ps1
Normal file
22
resources/scripts/Generate-Spinners.ps1
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
##########################################################
|
||||||
|
# Script that generates progress spinners.
|
||||||
|
##########################################################
|
||||||
|
|
||||||
|
$Output = Join-Path $PSScriptRoot "Temp"
|
||||||
|
$Source = Join-Path $PSScriptRoot "/../../src/Spectre.Console"
|
||||||
|
|
||||||
|
if(!(Test-Path $Output -PathType Container)) {
|
||||||
|
New-Item -ItemType Directory -Path $Output | Out-Null
|
||||||
|
}
|
||||||
|
|
||||||
|
# Generate the files
|
||||||
|
Push-Location Generator
|
||||||
|
&dotnet run -- spinners "$Output" --input $Output
|
||||||
|
if(!$?) {
|
||||||
|
Pop-Location
|
||||||
|
Throw "An error occured when generating code."
|
||||||
|
}
|
||||||
|
Pop-Location
|
||||||
|
|
||||||
|
# Copy the files to the correct location
|
||||||
|
Copy-Item (Join-Path "$Output" "Spinner.Generated.cs") -Destination "$Source/Progress/Spinner.Generated.cs"
|
||||||
@@ -7,7 +7,7 @@ using Spectre.IO;
|
|||||||
|
|
||||||
namespace Generator.Commands
|
namespace Generator.Commands
|
||||||
{
|
{
|
||||||
public sealed class ColorGeneratorCommand : Command<GeneratorCommandSettings>
|
public sealed class ColorGeneratorCommand : Command<ColorGeneratorCommand.Settings>
|
||||||
{
|
{
|
||||||
private readonly IFileSystem _fileSystem;
|
private readonly IFileSystem _fileSystem;
|
||||||
|
|
||||||
@@ -16,7 +16,13 @@ namespace Generator.Commands
|
|||||||
_fileSystem = new FileSystem();
|
_fileSystem = new FileSystem();
|
||||||
}
|
}
|
||||||
|
|
||||||
public override int Execute(CommandContext context, GeneratorCommandSettings settings)
|
public sealed class Settings : GeneratorSettings
|
||||||
|
{
|
||||||
|
[CommandOption("-i|--input <PATH>")]
|
||||||
|
public string Input { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public override int Execute(CommandContext context, Settings settings)
|
||||||
{
|
{
|
||||||
var templates = new FilePath[]
|
var templates = new FilePath[]
|
||||||
{
|
{
|
||||||
@@ -50,13 +56,4 @@ namespace Generator.Commands
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public sealed class GeneratorCommandSettings : CommandSettings
|
|
||||||
{
|
|
||||||
[CommandArgument(0, "<OUTPUT>")]
|
|
||||||
public string Output { get; set; }
|
|
||||||
|
|
||||||
[CommandOption("-i|--input <PATH>")]
|
|
||||||
public string Input { get; set; }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ using SpectreEnvironment = Spectre.IO.Environment;
|
|||||||
|
|
||||||
namespace Generator.Commands
|
namespace Generator.Commands
|
||||||
{
|
{
|
||||||
public sealed class EmojiGeneratorCommand : AsyncCommand<GeneratorCommandSettings>
|
public sealed class EmojiGeneratorCommand : AsyncCommand<EmojiGeneratorCommand.Settings>
|
||||||
{
|
{
|
||||||
private readonly IFileSystem _fileSystem;
|
private readonly IFileSystem _fileSystem;
|
||||||
private readonly IEnvironment _environment;
|
private readonly IEnvironment _environment;
|
||||||
@@ -24,9 +24,15 @@ namespace Generator.Commands
|
|||||||
private readonly Dictionary<string, string> _templates = new Dictionary<string, string>
|
private readonly Dictionary<string, string> _templates = new Dictionary<string, string>
|
||||||
{
|
{
|
||||||
{ "Templates/Emoji.Generated.template", "Emoji.Generated.cs" },
|
{ "Templates/Emoji.Generated.template", "Emoji.Generated.cs" },
|
||||||
{ "Templates/Emoji.Json.template", "emojis.json" },
|
{ "Templates/Emoji.Json.template", "emojis.json" }, // For documentation
|
||||||
};
|
};
|
||||||
|
|
||||||
|
public sealed class Settings : GeneratorSettings
|
||||||
|
{
|
||||||
|
[CommandOption("-i|--input <PATH>")]
|
||||||
|
public string Input { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
public EmojiGeneratorCommand()
|
public EmojiGeneratorCommand()
|
||||||
{
|
{
|
||||||
_fileSystem = new FileSystem();
|
_fileSystem = new FileSystem();
|
||||||
@@ -34,7 +40,7 @@ namespace Generator.Commands
|
|||||||
_parser = new HtmlParser();
|
_parser = new HtmlParser();
|
||||||
}
|
}
|
||||||
|
|
||||||
public override async Task<int> ExecuteAsync(CommandContext context, GeneratorCommandSettings settings)
|
public override async Task<int> ExecuteAsync(CommandContext context, Settings settings)
|
||||||
{
|
{
|
||||||
var output = new DirectoryPath(settings.Output);
|
var output = new DirectoryPath(settings.Output);
|
||||||
if (!_fileSystem.Directory.Exists(settings.Output))
|
if (!_fileSystem.Directory.Exists(settings.Output))
|
||||||
@@ -60,7 +66,7 @@ namespace Generator.Commands
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<Stream> FetchEmojis(GeneratorCommandSettings settings)
|
private async Task<Stream> FetchEmojis(Settings settings)
|
||||||
{
|
{
|
||||||
var input = string.IsNullOrEmpty(settings.Input)
|
var input = string.IsNullOrEmpty(settings.Input)
|
||||||
? _environment.WorkingDirectory
|
? _environment.WorkingDirectory
|
||||||
|
|||||||
10
resources/scripts/Generator/Commands/GeneratorSettings.cs
Normal file
10
resources/scripts/Generator/Commands/GeneratorSettings.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
using Spectre.Cli;
|
||||||
|
|
||||||
|
namespace Generator.Commands
|
||||||
|
{
|
||||||
|
public class GeneratorSettings : CommandSettings
|
||||||
|
{
|
||||||
|
[CommandArgument(0, "<OUTPUT>")]
|
||||||
|
public string Output { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using Generator.Models;
|
||||||
|
using Scriban;
|
||||||
|
using Spectre.Cli;
|
||||||
|
using Spectre.IO;
|
||||||
|
|
||||||
|
namespace Generator.Commands
|
||||||
|
{
|
||||||
|
public sealed class SpinnerGeneratorCommand : Command<GeneratorSettings>
|
||||||
|
{
|
||||||
|
private readonly IFileSystem _fileSystem;
|
||||||
|
|
||||||
|
public SpinnerGeneratorCommand()
|
||||||
|
{
|
||||||
|
_fileSystem = new FileSystem();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override int Execute(CommandContext context, GeneratorSettings settings)
|
||||||
|
{
|
||||||
|
// Read the spinner model.
|
||||||
|
var spinners = new List<Spinner>();
|
||||||
|
spinners.AddRange(Spinner.Parse(File.ReadAllText("Data/spinners_default.json")));
|
||||||
|
spinners.AddRange(Spinner.Parse(File.ReadAllText("Data/spinners_sindresorhus.json")));
|
||||||
|
|
||||||
|
var output = new DirectoryPath(settings.Output);
|
||||||
|
if (!_fileSystem.Directory.Exists(settings.Output))
|
||||||
|
{
|
||||||
|
_fileSystem.Directory.Create(settings.Output);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse the Scriban template.
|
||||||
|
var templatePath = new FilePath("Templates/Spinner.Generated.template");
|
||||||
|
var template = Template.Parse(File.ReadAllText(templatePath.FullPath));
|
||||||
|
|
||||||
|
// Render the template with the model.
|
||||||
|
var result = template.Render(new { Spinners = spinners });
|
||||||
|
|
||||||
|
// Write output to file
|
||||||
|
var file = output.CombineWithFilePath(templatePath.GetFilename().ChangeExtension(".cs"));
|
||||||
|
File.WriteAllText(file.FullPath, result);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
30
resources/scripts/Generator/Data/spinners_default.json
Normal file
30
resources/scripts/Generator/Data/spinners_default.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"Default": {
|
||||||
|
"interval": 100,
|
||||||
|
"unicode": true,
|
||||||
|
"frames": [
|
||||||
|
"⣷",
|
||||||
|
"⣯",
|
||||||
|
"⣟",
|
||||||
|
"⡿",
|
||||||
|
"⢿",
|
||||||
|
"⣻",
|
||||||
|
"⣽",
|
||||||
|
"⣾"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Ascii": {
|
||||||
|
"interval": 100,
|
||||||
|
"unicode": true,
|
||||||
|
"frames": [
|
||||||
|
"-",
|
||||||
|
"\\",
|
||||||
|
"|",
|
||||||
|
"/",
|
||||||
|
"-",
|
||||||
|
"\\",
|
||||||
|
"|",
|
||||||
|
"/"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
1368
resources/scripts/Generator/Data/spinners_sindresorhus.json
Normal file
1368
resources/scripts/Generator/Data/spinners_sindresorhus.json
Normal file
File diff suppressed because it is too large
Load Diff
@@ -15,6 +15,12 @@
|
|||||||
<None Update="Data\colors.json">
|
<None Update="Data\colors.json">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
|
<None Update="Data\spinners_default.json">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
<None Update="Data\spinners_sindresorhus.json">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
<None Update="Templates\ColorTable.Generated.template">
|
<None Update="Templates\ColorTable.Generated.template">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
@@ -24,6 +30,9 @@
|
|||||||
<None Update="Templates\ColorPalette.Generated.template">
|
<None Update="Templates\ColorPalette.Generated.template">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
|
<None Update="Templates\Spinner.Generated.template">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
<None Update="Templates\Emoji.Json.template">
|
<None Update="Templates\Emoji.Json.template">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
|
|||||||
31
resources/scripts/Generator/Models/Spinner.cs
Normal file
31
resources/scripts/Generator/Models/Spinner.cs
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using Humanizer;
|
||||||
|
using Newtonsoft.Json;
|
||||||
|
|
||||||
|
namespace Generator.Models
|
||||||
|
{
|
||||||
|
public sealed class Spinner
|
||||||
|
{
|
||||||
|
public string Name { get; set; }
|
||||||
|
public string NormalizedName { get; set; }
|
||||||
|
public int Interval { get; set; }
|
||||||
|
public bool Unicode { get; set; }
|
||||||
|
public List<string> Frames { get; set; }
|
||||||
|
|
||||||
|
public static IEnumerable<Spinner> Parse(string json)
|
||||||
|
{
|
||||||
|
var data = JsonConvert.DeserializeObject<Dictionary<string, Spinner>>(json);
|
||||||
|
foreach (var item in data)
|
||||||
|
{
|
||||||
|
item.Value.Name = item.Key;
|
||||||
|
item.Value.NormalizedName = item.Value.Name.Pascalize();
|
||||||
|
|
||||||
|
var frames = item.Value.Frames;
|
||||||
|
item.Value.Frames = frames.Select(f => f.Replace("\\", "\\\\")).ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
return data.Values;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -12,6 +12,7 @@ namespace Generator
|
|||||||
{
|
{
|
||||||
config.AddCommand<ColorGeneratorCommand>("colors");
|
config.AddCommand<ColorGeneratorCommand>("colors");
|
||||||
config.AddCommand<EmojiGeneratorCommand>("emoji");
|
config.AddCommand<EmojiGeneratorCommand>("emoji");
|
||||||
|
config.AddCommand<SpinnerGeneratorCommand>("spinners");
|
||||||
});
|
});
|
||||||
|
|
||||||
return app.Run(args);
|
return app.Run(args);
|
||||||
|
|||||||
@@ -0,0 +1,48 @@
|
|||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// <auto-generated>
|
||||||
|
// This code was generated by a tool.
|
||||||
|
// Generated {{ date.now | date.to_string `%F %R` }}
|
||||||
|
//
|
||||||
|
// Partly generated from
|
||||||
|
// https://github.com/sindresorhus/cli-spinners/blob/master/spinners.json
|
||||||
|
//
|
||||||
|
// Changes to this file may cause incorrect behavior and will be lost if
|
||||||
|
// the code is regenerated.
|
||||||
|
// </auto-generated>
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
public abstract partial class Spinner
|
||||||
|
{
|
||||||
|
{{~ for spinner in spinners ~}}
|
||||||
|
private sealed class {{ spinner.normalized_name }}Spinner : Spinner
|
||||||
|
{
|
||||||
|
public override TimeSpan Interval => TimeSpan.FromMilliseconds({{ spinner.interval }});
|
||||||
|
public override bool IsUnicode => {{ spinner.unicode }};
|
||||||
|
public override IReadOnlyList<string> Frames => new List<string>
|
||||||
|
{
|
||||||
|
{{~ for frame in spinner.frames ~}}
|
||||||
|
"{{ frame }}",
|
||||||
|
{{~ end ~}}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
{{~ end ~}}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Contains all predefined spinners.
|
||||||
|
/// </summary>
|
||||||
|
public static class Known
|
||||||
|
{
|
||||||
|
{{~ for spinner in spinners ~}}
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the "{{ spinner.name }}" spinner.
|
||||||
|
/// </summary>
|
||||||
|
public static Spinner {{ spinner.normalized_name }} { get; } = new {{ spinner.normalized_name }}Spinner();
|
||||||
|
{{~ end ~}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -89,4 +89,7 @@ dotnet_diagnostic.RCS1227.severity = none
|
|||||||
dotnet_diagnostic.IDE0004.severity = warning
|
dotnet_diagnostic.IDE0004.severity = warning
|
||||||
|
|
||||||
# CA1810: Initialize reference type static fields inline
|
# CA1810: Initialize reference type static fields inline
|
||||||
dotnet_diagnostic.CA1810.severity = none
|
dotnet_diagnostic.CA1810.severity = none
|
||||||
|
|
||||||
|
# IDE0044: Add readonly modifier
|
||||||
|
dotnet_diagnostic.IDE0044.severity = warning
|
||||||
125
src/Spectre.Console.ImageSharp/CanvasImage.cs
Normal file
125
src/Spectre.Console.ImageSharp/CanvasImage.cs
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using SixLabors.ImageSharp.PixelFormats;
|
||||||
|
using SixLabors.ImageSharp.Processing;
|
||||||
|
using SixLabors.ImageSharp.Processing.Processors.Transforms;
|
||||||
|
using Spectre.Console.Rendering;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Represents a renderable image.
|
||||||
|
/// </summary>
|
||||||
|
public sealed class CanvasImage : Renderable
|
||||||
|
{
|
||||||
|
private static readonly IResampler _defaultResampler = KnownResamplers.Bicubic;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the image width.
|
||||||
|
/// </summary>
|
||||||
|
public int Width => Image.Width;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the image height.
|
||||||
|
/// </summary>
|
||||||
|
public int Height => Image.Height;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the render width of the canvas.
|
||||||
|
/// </summary>
|
||||||
|
public int? MaxWidth { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the render width of the canvas.
|
||||||
|
/// </summary>
|
||||||
|
public int PixelWidth { get; set; } = 2;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the <see cref="IResampler"/> that should
|
||||||
|
/// be used when scaling the image. Defaults to bicubic sampling.
|
||||||
|
/// </summary>
|
||||||
|
public IResampler? Resampler { get; set; }
|
||||||
|
|
||||||
|
internal SixLabors.ImageSharp.Image<Rgba32> Image { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="CanvasImage"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="filename">The image filename.</param>
|
||||||
|
public CanvasImage(string filename)
|
||||||
|
{
|
||||||
|
Image = SixLabors.ImageSharp.Image.Load<Rgba32>(filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
protected override Measurement Measure(RenderContext context, int maxWidth)
|
||||||
|
{
|
||||||
|
if (PixelWidth < 0)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Pixel width must be greater than zero.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var width = MaxWidth ?? Width;
|
||||||
|
if (maxWidth < width * PixelWidth)
|
||||||
|
{
|
||||||
|
return new Measurement(maxWidth, maxWidth);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Measurement(width * PixelWidth, width * PixelWidth);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
protected override IEnumerable<Segment> Render(RenderContext context, int maxWidth)
|
||||||
|
{
|
||||||
|
var image = Image;
|
||||||
|
|
||||||
|
var width = Width;
|
||||||
|
var height = Height;
|
||||||
|
|
||||||
|
// Got a max width?
|
||||||
|
if (MaxWidth != null)
|
||||||
|
{
|
||||||
|
height = (int)(height * ((float)MaxWidth.Value) / Width);
|
||||||
|
width = MaxWidth.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exceed the max width when we take pixel width into account?
|
||||||
|
if (width * PixelWidth > maxWidth)
|
||||||
|
{
|
||||||
|
height = (int)(height * (maxWidth / (float)(width * PixelWidth)));
|
||||||
|
width = maxWidth / PixelWidth;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Need to rescale the pixel buffer?
|
||||||
|
if (width != Width || height != Height)
|
||||||
|
{
|
||||||
|
var resampler = Resampler ?? _defaultResampler;
|
||||||
|
image = image.Clone(); // Clone the original image
|
||||||
|
image.Mutate(i => i.Resize(width, height, resampler));
|
||||||
|
}
|
||||||
|
|
||||||
|
var canvas = new Canvas(width, height)
|
||||||
|
{
|
||||||
|
MaxWidth = MaxWidth,
|
||||||
|
PixelWidth = PixelWidth,
|
||||||
|
Scale = false,
|
||||||
|
};
|
||||||
|
|
||||||
|
for (var y = 0; y < image.Height; y++)
|
||||||
|
{
|
||||||
|
for (var x = 0; x < image.Width; x++)
|
||||||
|
{
|
||||||
|
if (image[x, y].A == 0)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
canvas.SetPixel(x, y, new Color(
|
||||||
|
image[x, y].R, image[x, y].G, image[x, y].B));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ((IRenderable)canvas).Render(context, maxWidth);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
135
src/Spectre.Console.ImageSharp/CanvasImageExtensions.cs
Normal file
135
src/Spectre.Console.ImageSharp/CanvasImageExtensions.cs
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
using System;
|
||||||
|
using SixLabors.ImageSharp.Processing;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="CanvasImage"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class CanvasImageExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the maximum width of the rendered image.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <param name="maxWidth">The maximum width.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage MaxWidth(this CanvasImage image, int? maxWidth)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.MaxWidth = maxWidth;
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Disables the maximum width of the rendered image.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage NoMaxWidth(this CanvasImage image)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.MaxWidth = null;
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the pixel width.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <param name="width">The pixel width.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage PixelWidth(this CanvasImage image, int width)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.PixelWidth = width;
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Mutates the underlying image.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <param name="action">The action that mutates the underlying image.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage Mutate(this CanvasImage image, Action<IImageProcessingContext> action)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (action is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(action));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.Image.Mutate(action);
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Uses a bicubic sampler that implements the bicubic kernel algorithm W(x).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage BicubicResampler(this CanvasImage image)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.Resampler = KnownResamplers.Bicubic;
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Uses a bilinear sampler. This interpolation algorithm
|
||||||
|
/// can be used where perfect image transformation with pixel matching is impossible,
|
||||||
|
/// so that one can calculate and assign appropriate intensity values to pixels.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage BilinearResampler(this CanvasImage image)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.Resampler = KnownResamplers.Triangle;
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Uses a Nearest-Neighbour sampler that implements the nearest neighbor algorithm.
|
||||||
|
/// This uses a very fast, unscaled filter which will select the closest pixel to
|
||||||
|
/// the new pixels position.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="image">The canvas image.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static CanvasImage NearestNeighborResampler(this CanvasImage image)
|
||||||
|
{
|
||||||
|
if (image is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(image));
|
||||||
|
}
|
||||||
|
|
||||||
|
image.Resampler = KnownResamplers.NearestNeighbor;
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFrameworks>netstandard2.0</TargetFrameworks>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<Description>A library that extends Spectre.Console with ImageSharp superpowers.</Description>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<AdditionalFiles Include="..\stylecop.json" Link="Properties/stylecop.json" />
|
||||||
|
<None Include="../../resources/gfx/small-logo.png" Pack="true" PackagePath="\" Link="Properties/small-logo.png" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="SixLabors.ImageSharp" Version="1.0.2" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Spectre.Console\Spectre.Console.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -8,6 +8,9 @@ namespace Spectre.Console.Tests.Data
|
|||||||
[SuppressMessage("Usage", "CA1801:Review unused parameters", Justification = "<Pending>")]
|
[SuppressMessage("Usage", "CA1801:Review unused parameters", Justification = "<Pending>")]
|
||||||
public static bool MethodThatThrows(int? number) => throw new InvalidOperationException("Throwing!");
|
public static bool MethodThatThrows(int? number) => throw new InvalidOperationException("Throwing!");
|
||||||
|
|
||||||
|
[SuppressMessage("Usage", "CA1801:Review unused parameters", Justification = "<Pending>")]
|
||||||
|
public static bool GenericMethodThatThrows<T0, T1, TRet>(int? number) => throw new InvalidOperationException("Throwing!");
|
||||||
|
|
||||||
public static void ThrowWithInnerException()
|
public static void ThrowWithInnerException()
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@@ -19,5 +22,17 @@ namespace Spectre.Console.Tests.Data
|
|||||||
throw new InvalidOperationException("Something threw!", ex);
|
throw new InvalidOperationException("Something threw!", ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void ThrowWithGenericInnerException()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
GenericMethodThatThrows<int, float, double>(null);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Something threw!", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
System.InvalidOperationException: Something threw!
|
||||||
|
System.InvalidOperationException: Throwing!
|
||||||
|
at Spectre.Console.Tests.Data.TestExceptions.GenericMethodThatThrows[[T0,T1,TRet]](Nullable`1 number) in /xyz/Exceptions.cs:nn
|
||||||
|
at Spectre.Console.Tests.Data.TestExceptions.ThrowWithGenericInnerException() in /xyz/Exceptions.cs:nn
|
||||||
|
at Spectre.Console.Tests.Data.TestExceptions.ThrowWithGenericInnerException() in /xyz/Exceptions.cs:nn
|
||||||
|
at Spectre.Console.Tests.Unit.ExceptionTests.<>c.<Should_Write_Exceptions_With_Generic_Type_Parameters_In_Callsite_As_Expected>b__4_0() in /xyz/ExceptionTests.cs:nn
|
||||||
|
at Spectre.Console.Tests.Unit.ExceptionTests.GetException(Action action) in /xyz/ExceptionTests.cs:nn
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
┌──────────┐
|
||||||
|
│ ┌──────┐ │
|
||||||
|
│ │ 测试 │ │
|
||||||
|
│ ├──────┤ │
|
||||||
|
│ │ 测试 │ │
|
||||||
|
│ └──────┘ │
|
||||||
|
└──────────┘
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
━━━━━━━━━━━━━━━━━━━━
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
|
||||||
|
foo ━━━ 0% -:--:-- ⣷
|
||||||
|
bar ━━━ 0% -:--:-- ⣷
|
||||||
|
baz ━━━ 0% -:--:-- ⣷
|
||||||
|
|
||||||
17
src/Spectre.Console.Tests/Tools/DummyCursor.cs
Normal file
17
src/Spectre.Console.Tests/Tools/DummyCursor.cs
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
namespace Spectre.Console.Tests
|
||||||
|
{
|
||||||
|
public sealed class DummyCursor : IAnsiConsoleCursor
|
||||||
|
{
|
||||||
|
public void Move(CursorDirection direction, int steps)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetPosition(int column, int line)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Show(bool show)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
25
src/Spectre.Console.Tests/Tools/DummySpinners.cs
Normal file
25
src/Spectre.Console.Tests/Tools/DummySpinners.cs
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace Spectre.Console.Tests
|
||||||
|
{
|
||||||
|
public sealed class DummySpinner1 : Spinner
|
||||||
|
{
|
||||||
|
public override TimeSpan Interval => TimeSpan.FromMilliseconds(100);
|
||||||
|
public override bool IsUnicode => true;
|
||||||
|
public override IReadOnlyList<string> Frames => new List<string>
|
||||||
|
{
|
||||||
|
"*",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class DummySpinner2 : Spinner
|
||||||
|
{
|
||||||
|
public override TimeSpan Interval => TimeSpan.FromMilliseconds(100);
|
||||||
|
public override bool IsUnicode => true;
|
||||||
|
public override IReadOnlyList<string> Frames => new List<string>
|
||||||
|
{
|
||||||
|
"-",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -11,13 +11,14 @@ namespace Spectre.Console.Tests
|
|||||||
{
|
{
|
||||||
public Capabilities Capabilities { get; }
|
public Capabilities Capabilities { get; }
|
||||||
public Encoding Encoding { get; }
|
public Encoding Encoding { get; }
|
||||||
public IAnsiConsoleCursor Cursor => throw new NotSupportedException();
|
public IAnsiConsoleCursor Cursor => new DummyCursor();
|
||||||
public TestableConsoleInput Input { get; }
|
public TestableConsoleInput Input { get; }
|
||||||
|
|
||||||
public int Width { get; }
|
public int Width { get; }
|
||||||
public int Height { get; }
|
public int Height { get; }
|
||||||
|
|
||||||
IAnsiConsoleInput IAnsiConsole.Input => Input;
|
IAnsiConsoleInput IAnsiConsole.Input => Input;
|
||||||
|
public RenderPipeline Pipeline { get; }
|
||||||
|
|
||||||
public Decoration Decoration { get; set; }
|
public Decoration Decoration { get; set; }
|
||||||
public Color Foreground { get; set; }
|
public Color Foreground { get; set; }
|
||||||
@@ -31,14 +32,15 @@ namespace Spectre.Console.Tests
|
|||||||
public PlainConsole(
|
public PlainConsole(
|
||||||
int width = 80, int height = 9000, Encoding encoding = null,
|
int width = 80, int height = 9000, Encoding encoding = null,
|
||||||
bool supportsAnsi = true, ColorSystem colorSystem = ColorSystem.Standard,
|
bool supportsAnsi = true, ColorSystem colorSystem = ColorSystem.Standard,
|
||||||
bool legacyConsole = false)
|
bool legacyConsole = false, bool interactive = true)
|
||||||
{
|
{
|
||||||
Capabilities = new Capabilities(supportsAnsi, colorSystem, legacyConsole);
|
Capabilities = new Capabilities(supportsAnsi, colorSystem, legacyConsole, interactive);
|
||||||
Encoding = encoding ?? Encoding.UTF8;
|
Encoding = encoding ?? Encoding.UTF8;
|
||||||
Width = width;
|
Width = width;
|
||||||
Height = height;
|
Height = height;
|
||||||
Writer = new StringWriter();
|
Writer = new StringWriter();
|
||||||
Input = new TestableConsoleInput();
|
Input = new TestableConsoleInput();
|
||||||
|
Pipeline = new RenderPipeline();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Dispose()
|
public void Dispose()
|
||||||
@@ -50,14 +52,17 @@ namespace Spectre.Console.Tests
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Write(Segment segment)
|
public void Write(IEnumerable<Segment> segments)
|
||||||
{
|
{
|
||||||
if (segment is null)
|
if (segments is null)
|
||||||
{
|
{
|
||||||
throw new ArgumentNullException(nameof(segment));
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Writer.Write(segment.Text);
|
foreach (var segment in segments)
|
||||||
|
{
|
||||||
|
Writer.Write(segment.Text);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public string WriteNormalizedException(Exception ex, ExceptionFormats formats = ExceptionFormats.Default)
|
public string WriteNormalizedException(Exception ex, ExceptionFormats formats = ExceptionFormats.Default)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
namespace Spectre.Console.Tests.Tools
|
namespace Spectre.Console.Tests
|
||||||
{
|
{
|
||||||
public sealed class TestLinkIdentityGenerator : ILinkIdentityGenerator
|
public sealed class TestLinkIdentityGenerator : ILinkIdentityGenerator
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using Spectre.Console.Rendering;
|
using Spectre.Console.Rendering;
|
||||||
using Spectre.Console.Tests.Tools;
|
|
||||||
|
|
||||||
namespace Spectre.Console.Tests
|
namespace Spectre.Console.Tests
|
||||||
{
|
{
|
||||||
@@ -19,16 +19,21 @@ namespace Spectre.Console.Tests
|
|||||||
public int Height => _console.Height;
|
public int Height => _console.Height;
|
||||||
public IAnsiConsoleCursor Cursor => _console.Cursor;
|
public IAnsiConsoleCursor Cursor => _console.Cursor;
|
||||||
public TestableConsoleInput Input { get; }
|
public TestableConsoleInput Input { get; }
|
||||||
|
public RenderPipeline Pipeline => _console.Pipeline;
|
||||||
|
|
||||||
IAnsiConsoleInput IAnsiConsole.Input => Input;
|
IAnsiConsoleInput IAnsiConsole.Input => Input;
|
||||||
|
|
||||||
public TestableAnsiConsole(ColorSystem system, AnsiSupport ansi = AnsiSupport.Yes, int width = 80)
|
public TestableAnsiConsole(
|
||||||
|
ColorSystem system, AnsiSupport ansi = AnsiSupport.Yes,
|
||||||
|
InteractionSupport interaction = InteractionSupport.Yes,
|
||||||
|
int width = 80)
|
||||||
{
|
{
|
||||||
_writer = new StringWriter();
|
_writer = new StringWriter();
|
||||||
_console = AnsiConsole.Create(new AnsiConsoleSettings
|
_console = AnsiConsole.Create(new AnsiConsoleSettings
|
||||||
{
|
{
|
||||||
Ansi = ansi,
|
Ansi = ansi,
|
||||||
ColorSystem = (ColorSystemSupport)system,
|
ColorSystem = (ColorSystemSupport)system,
|
||||||
|
Interactive = interaction,
|
||||||
Out = _writer,
|
Out = _writer,
|
||||||
LinkIdentityGenerator = new TestLinkIdentityGenerator(),
|
LinkIdentityGenerator = new TestLinkIdentityGenerator(),
|
||||||
});
|
});
|
||||||
@@ -47,9 +52,17 @@ namespace Spectre.Console.Tests
|
|||||||
_console.Clear(home);
|
_console.Clear(home);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Write(Segment segment)
|
public void Write(IEnumerable<Segment> segments)
|
||||||
{
|
{
|
||||||
_console.Write(segment);
|
if (segments is null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var segment in segments)
|
||||||
|
{
|
||||||
|
_console.Write(segment);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -65,6 +65,20 @@ namespace Spectre.Console.Tests.Unit
|
|||||||
return Verifier.Verify(result);
|
return Verifier.Verify(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public Task Should_Write_Exceptions_With_Generic_Type_Parameters_In_Callsite_As_Expected()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new PlainConsole(width: 1024);
|
||||||
|
var dex = GetException(() => TestExceptions.ThrowWithGenericInnerException());
|
||||||
|
|
||||||
|
// When
|
||||||
|
var result = console.WriteNormalizedException(dex);
|
||||||
|
|
||||||
|
// Then
|
||||||
|
return Verifier.Verify(result);
|
||||||
|
}
|
||||||
|
|
||||||
public static Exception GetException(Action action)
|
public static Exception GetException(Action action)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using Shouldly;
|
||||||
using Spectre.Console.Rendering;
|
using Spectre.Console.Rendering;
|
||||||
using VerifyXunit;
|
using VerifyXunit;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
@@ -267,5 +269,23 @@ namespace Spectre.Console.Tests.Unit
|
|||||||
// Then
|
// Then
|
||||||
return Verifier.Verify(console.Output);
|
return Verifier.Verify(console.Output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public Task Should_Wrap_Table_With_CJK_Tables_In_Panel_Correctly()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new PlainConsole(width: 80);
|
||||||
|
|
||||||
|
var table = new Table();
|
||||||
|
table.AddColumn("测试");
|
||||||
|
table.AddRow("测试");
|
||||||
|
var panel = new Panel(table);
|
||||||
|
|
||||||
|
// When
|
||||||
|
console.Render(panel);
|
||||||
|
|
||||||
|
// Then
|
||||||
|
return Verifier.Verify(console.Output);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
115
src/Spectre.Console.Tests/Unit/ProgressTests.cs
Normal file
115
src/Spectre.Console.Tests/Unit/ProgressTests.cs
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using Shouldly;
|
||||||
|
using VerifyXunit;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Spectre.Console.Tests.Unit
|
||||||
|
{
|
||||||
|
[UsesVerify]
|
||||||
|
public sealed class ProgressTests
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public void Should_Render_Task_Correctly()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new TestableAnsiConsole(ColorSystem.TrueColor, width: 10);
|
||||||
|
|
||||||
|
var progress = new Progress(console)
|
||||||
|
.Columns(new[] { new ProgressBarColumn() })
|
||||||
|
.AutoRefresh(false)
|
||||||
|
.AutoClear(true);
|
||||||
|
|
||||||
|
// When
|
||||||
|
progress.Start(ctx => ctx.AddTask("foo"));
|
||||||
|
|
||||||
|
// Then
|
||||||
|
console.Output
|
||||||
|
.NormalizeLineEndings()
|
||||||
|
.ShouldBe(
|
||||||
|
"[?25l" + // Hide cursor
|
||||||
|
" \n" + // Top padding
|
||||||
|
"[38;5;8m━━━━━━━━━━[0m\n" + // Task
|
||||||
|
" " + // Bottom padding
|
||||||
|
"[2K[1A[2K[1A[2K[?25h"); // Clear + show cursor
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Should_Not_Auto_Clear_If_Specified()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new TestableAnsiConsole(ColorSystem.TrueColor, width: 10);
|
||||||
|
|
||||||
|
var progress = new Progress(console)
|
||||||
|
.Columns(new[] { new ProgressBarColumn() })
|
||||||
|
.AutoRefresh(false)
|
||||||
|
.AutoClear(false);
|
||||||
|
|
||||||
|
// When
|
||||||
|
progress.Start(ctx => ctx.AddTask("foo"));
|
||||||
|
|
||||||
|
// Then
|
||||||
|
console.Output
|
||||||
|
.NormalizeLineEndings()
|
||||||
|
.ShouldBe(
|
||||||
|
"[?25l" + // Hide cursor
|
||||||
|
" \n" + // Top padding
|
||||||
|
"[38;5;8m━━━━━━━━━━[0m\n" + // Task
|
||||||
|
" \n" + // Bottom padding
|
||||||
|
"[?25h"); // show cursor
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public Task Should_Reduce_Width_If_Needed()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new PlainConsole(width: 20);
|
||||||
|
|
||||||
|
var progress = new Progress(console)
|
||||||
|
.Columns(new ProgressColumn[]
|
||||||
|
{
|
||||||
|
new TaskDescriptionColumn(),
|
||||||
|
new ProgressBarColumn(),
|
||||||
|
new PercentageColumn(),
|
||||||
|
new RemainingTimeColumn(),
|
||||||
|
new SpinnerColumn(),
|
||||||
|
})
|
||||||
|
.AutoRefresh(false)
|
||||||
|
.AutoClear(false);
|
||||||
|
|
||||||
|
// When
|
||||||
|
progress.Start(ctx =>
|
||||||
|
{
|
||||||
|
ctx.AddTask("foo");
|
||||||
|
ctx.AddTask("bar");
|
||||||
|
ctx.AddTask("baz");
|
||||||
|
});
|
||||||
|
|
||||||
|
// Then
|
||||||
|
return Verifier.Verify(console.Output);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Setting_Max_Value_Should_Set_The_MaxValue_And_Cap_Value()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var task = default(ProgressTask);
|
||||||
|
var console = new PlainConsole();
|
||||||
|
var progress = new Progress(console)
|
||||||
|
.Columns(new[] { new ProgressBarColumn() })
|
||||||
|
.AutoRefresh(false)
|
||||||
|
.AutoClear(false);
|
||||||
|
|
||||||
|
// When
|
||||||
|
progress.Start(ctx =>
|
||||||
|
{
|
||||||
|
task = ctx.AddTask("foo");
|
||||||
|
task.Increment(100);
|
||||||
|
task.MaxValue = 20;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Then
|
||||||
|
task.MaxValue.ShouldBe(20);
|
||||||
|
task.Value.ShouldBe(20);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
34
src/Spectre.Console.Tests/Unit/RenderHookTests.cs
Normal file
34
src/Spectre.Console.Tests/Unit/RenderHookTests.cs
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using Shouldly;
|
||||||
|
using Spectre.Console.Rendering;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Spectre.Console.Tests.Unit
|
||||||
|
{
|
||||||
|
public sealed class RenderHookTests
|
||||||
|
{
|
||||||
|
private sealed class HelloRenderHook : IRenderHook
|
||||||
|
{
|
||||||
|
public IEnumerable<IRenderable> Process(RenderContext context, IEnumerable<IRenderable> renderables)
|
||||||
|
{
|
||||||
|
return new IRenderable[] { new Text("Hello\n") }.Concat(renderables);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Should_Inject_Renderable_Before_Writing_To_Console()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new PlainConsole();
|
||||||
|
console.Pipeline.Attach(new HelloRenderHook());
|
||||||
|
|
||||||
|
// When
|
||||||
|
console.Render(new Text("World"));
|
||||||
|
|
||||||
|
// Then
|
||||||
|
console.Lines[0].ShouldBe("Hello");
|
||||||
|
console.Lines[1].ShouldBe("World");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -22,18 +22,43 @@ namespace Spectre.Console.Tests.Unit
|
|||||||
[UsesVerify]
|
[UsesVerify]
|
||||||
public sealed class TheSplitMethod
|
public sealed class TheSplitMethod
|
||||||
{
|
{
|
||||||
[Fact]
|
[Theory]
|
||||||
public Task Should_Split_Segment_Correctly()
|
[InlineData("Foo Bar", 0, "", "Foo Bar")]
|
||||||
|
[InlineData("Foo Bar", 1, "F", "oo Bar")]
|
||||||
|
[InlineData("Foo Bar", 2, "Fo", "o Bar")]
|
||||||
|
[InlineData("Foo Bar", 3, "Foo", " Bar")]
|
||||||
|
[InlineData("Foo Bar", 4, "Foo ", "Bar")]
|
||||||
|
[InlineData("Foo Bar", 5, "Foo B", "ar")]
|
||||||
|
[InlineData("Foo Bar", 6, "Foo Ba", "r")]
|
||||||
|
[InlineData("Foo Bar", 7, "Foo Bar", null)]
|
||||||
|
[InlineData("Foo 测试 Bar", 0, "", "Foo 测试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 1, "F", "oo 测试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 2, "Fo", "o 测试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 3, "Foo", " 测试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 4, "Foo ", "测试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 5, "Foo 测", "试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 6, "Foo 测", "试 Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 7, "Foo 测试", " Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 8, "Foo 测试", " Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 9, "Foo 测试 ", "Bar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 10, "Foo 测试 B", "ar")]
|
||||||
|
[InlineData("Foo 测试 Bar", 11, "Foo 测试 Ba", "r")]
|
||||||
|
[InlineData("Foo 测试 Bar", 12, "Foo 测试 Bar", null)]
|
||||||
|
public void Should_Split_Segment_Correctly(string text, int offset, string expectedFirst, string expectedSecond)
|
||||||
{
|
{
|
||||||
// Given
|
// Given
|
||||||
var style = new Style(Color.Red, Color.Green, Decoration.Bold);
|
var style = new Style(Color.Red, Color.Green, Decoration.Bold);
|
||||||
var segment = new Segment("Foo Bar", style);
|
var context = new RenderContext(Encoding.UTF8, false);
|
||||||
|
var segment = new Segment(text, style);
|
||||||
|
|
||||||
// When
|
// When
|
||||||
var result = segment.Split(3);
|
var (first, second) = segment.Split(context, offset);
|
||||||
|
|
||||||
// Then
|
// Then
|
||||||
return Verifier.Verify(result);
|
first.Text.ShouldBe(expectedFirst);
|
||||||
|
first.Style.ShouldBe(style);
|
||||||
|
second?.Text?.ShouldBe(expectedSecond);
|
||||||
|
second?.Style?.ShouldBe(style);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
42
src/Spectre.Console.Tests/Unit/StatusTests.cs
Normal file
42
src/Spectre.Console.Tests/Unit/StatusTests.cs
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
using Shouldly;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Spectre.Console.Tests.Unit
|
||||||
|
{
|
||||||
|
public sealed partial class StatusTests
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public void Should_Render_Status_Correctly()
|
||||||
|
{
|
||||||
|
// Given
|
||||||
|
var console = new TestableAnsiConsole(ColorSystem.TrueColor, width: 10);
|
||||||
|
|
||||||
|
var status = new Status(console);
|
||||||
|
status.AutoRefresh = false;
|
||||||
|
status.Spinner = new DummySpinner1();
|
||||||
|
|
||||||
|
// When
|
||||||
|
status.Start("foo", ctx =>
|
||||||
|
{
|
||||||
|
ctx.Refresh();
|
||||||
|
ctx.Spinner(new DummySpinner2());
|
||||||
|
ctx.Status("bar");
|
||||||
|
ctx.Refresh();
|
||||||
|
ctx.Spinner(new DummySpinner1());
|
||||||
|
ctx.Status("baz");
|
||||||
|
});
|
||||||
|
|
||||||
|
// Then
|
||||||
|
console.Output
|
||||||
|
.NormalizeLineEndings()
|
||||||
|
.ShouldBe(
|
||||||
|
"[?25l \n" +
|
||||||
|
"[38;5;11m*[0m foo\n" +
|
||||||
|
" [1A[1A \n" +
|
||||||
|
"[38;5;11m-[0m bar\n" +
|
||||||
|
" [1A[1A \n" +
|
||||||
|
"[38;5;11m*[0m baz\n" +
|
||||||
|
" [2K[1A[2K[1A[2K[?25h");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -54,6 +54,14 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Prompt", "..\examples\Promp
|
|||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Figlet", "..\examples\Figlet\Figlet.csproj", "{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}"
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Figlet", "..\examples\Figlet\Figlet.csproj", "{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Canvas", "..\examples\Canvas\Canvas.csproj", "{5693761A-754A-40A8-9144-36510D6A4D69}"
|
||||||
|
EndProject
|
||||||
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Spectre.Console.ImageSharp", "Spectre.Console.ImageSharp\Spectre.Console.ImageSharp.csproj", "{0EFE694D-0770-4E71-BF4E-EC2B41362F79}"
|
||||||
|
EndProject
|
||||||
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Progress", "..\examples\Progress\Progress.csproj", "{2B712A52-40F1-4C1C-833E-7C869ACA91F3}"
|
||||||
|
EndProject
|
||||||
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Status", "..\examples\Status\Status.csproj", "{3716AFDF-0904-4635-8422-86E6B9356840}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
@@ -268,6 +276,54 @@ Global
|
|||||||
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}.Release|x64.Build.0 = Release|Any CPU
|
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}.Release|x64.Build.0 = Release|Any CPU
|
||||||
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}.Release|x86.ActiveCfg = Release|Any CPU
|
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}.Release|x86.ActiveCfg = Release|Any CPU
|
||||||
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}.Release|x86.Build.0 = Release|Any CPU
|
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1}.Release|x86.Build.0 = Release|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Debug|x64.Build.0 = Debug|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Debug|x86.Build.0 = Debug|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Release|x64.ActiveCfg = Release|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Release|x64.Build.0 = Release|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Release|x86.ActiveCfg = Release|Any CPU
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69}.Release|x86.Build.0 = Release|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Debug|x64.Build.0 = Debug|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Debug|x86.Build.0 = Debug|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Release|x64.ActiveCfg = Release|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Release|x64.Build.0 = Release|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Release|x86.ActiveCfg = Release|Any CPU
|
||||||
|
{0EFE694D-0770-4E71-BF4E-EC2B41362F79}.Release|x86.Build.0 = Release|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Debug|x64.Build.0 = Debug|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Debug|x86.Build.0 = Debug|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Release|x64.ActiveCfg = Release|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Release|x64.Build.0 = Release|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Release|x86.ActiveCfg = Release|Any CPU
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3}.Release|x86.Build.0 = Release|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Debug|x64.Build.0 = Debug|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Debug|x86.Build.0 = Debug|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Release|x64.ActiveCfg = Release|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Release|x64.Build.0 = Release|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Release|x86.ActiveCfg = Release|Any CPU
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840}.Release|x86.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
@@ -289,6 +345,9 @@ Global
|
|||||||
{75C608C3-ABB4-4168-A229-7F8250B946D1} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
{75C608C3-ABB4-4168-A229-7F8250B946D1} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
||||||
{6351C70F-F368-46DB-BAED-9B87CCD69353} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
{6351C70F-F368-46DB-BAED-9B87CCD69353} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
||||||
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
{45BF6302-6553-4E52-BF0F-B10D1AA9A6D1} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
||||||
|
{5693761A-754A-40A8-9144-36510D6A4D69} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
||||||
|
{2B712A52-40F1-4C1C-833E-7C869ACA91F3} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
||||||
|
{3716AFDF-0904-4635-8422-86E6B9356840} = {F0575243-121F-4DEE-9F6B-246E26DC0844}
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||||
SolutionGuid = {5729B071-67A0-48FB-8B1B-275E6822086C}
|
SolutionGuid = {5729B071-67A0-48FB-8B1B-275E6822086C}
|
||||||
|
|||||||
26
src/Spectre.Console/AnsiConsole.Progress.cs
Normal file
26
src/Spectre.Console/AnsiConsole.Progress.cs
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// A console capable of writing ANSI escape sequences.
|
||||||
|
/// </summary>
|
||||||
|
public static partial class AnsiConsole
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a new <see cref="Progress"/> instance.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A <see cref="Progress"/> instance.</returns>
|
||||||
|
public static Progress Progress()
|
||||||
|
{
|
||||||
|
return Console.Progress();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a new <see cref="Status"/> instance.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A <see cref="Status"/> instance.</returns>
|
||||||
|
public static Status Status()
|
||||||
|
{
|
||||||
|
return Console.Status();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -18,6 +18,12 @@ namespace Spectre.Console
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public ColorSystemSupport ColorSystem { get; set; }
|
public ColorSystemSupport ColorSystem { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a value indicating whether or
|
||||||
|
/// not the console is interactive.
|
||||||
|
/// </summary>
|
||||||
|
public InteractionSupport Interactive { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the link identity generator.
|
/// Gets or sets the link identity generator.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -36,17 +36,24 @@ namespace Spectre.Console
|
|||||||
/// </remarks>
|
/// </remarks>
|
||||||
public bool LegacyConsole { get; }
|
public bool LegacyConsole { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a value indicating whether or not the console supports interaction.
|
||||||
|
/// </summary>
|
||||||
|
public bool SupportsInteraction { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="Capabilities"/> class.
|
/// Initializes a new instance of the <see cref="Capabilities"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="supportsAnsi">Whether or not ANSI escape sequences are supported.</param>
|
/// <param name="supportsAnsi">Whether or not ANSI escape sequences are supported.</param>
|
||||||
/// <param name="colorSystem">The color system that is supported.</param>
|
/// <param name="colorSystem">The color system that is supported.</param>
|
||||||
/// <param name="legacyConsole">Whether or not this is a legacy console.</param>
|
/// <param name="legacyConsole">Whether or not this is a legacy console.</param>
|
||||||
public Capabilities(bool supportsAnsi, ColorSystem colorSystem, bool legacyConsole)
|
/// <param name="supportsInteraction">Whether or not the console supports interaction.</param>
|
||||||
|
public Capabilities(bool supportsAnsi, ColorSystem colorSystem, bool legacyConsole, bool supportsInteraction)
|
||||||
{
|
{
|
||||||
SupportsAnsi = supportsAnsi;
|
SupportsAnsi = supportsAnsi;
|
||||||
ColorSystem = colorSystem;
|
ColorSystem = colorSystem;
|
||||||
LegacyConsole = legacyConsole;
|
LegacyConsole = legacyConsole;
|
||||||
|
SupportsInteraction = supportsInteraction;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -8,31 +8,31 @@ namespace Spectre.Console
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Try to detect the color system.
|
/// Try to detect the color system.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
Detect = -1,
|
Detect = 0,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// No colors.
|
/// No colors.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
NoColors = 0,
|
NoColors = 1,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Legacy, 3-bit mode.
|
/// Legacy, 3-bit mode.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
Legacy = 1,
|
Legacy = 2,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Standard, 4-bit mode.
|
/// Standard, 4-bit mode.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
Standard = 2,
|
Standard = 3,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 8-bit mode.
|
/// 8-bit mode.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
EightBit = 3,
|
EightBit = 4,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 24-bit mode.
|
/// 24-bit mode.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
TrueColor = 4,
|
TrueColor = 5,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,8 +52,7 @@ namespace Spectre.Console
|
|||||||
/// <param name="args">An array of objects to write.</param>
|
/// <param name="args">An array of objects to write.</param>
|
||||||
public static void MarkupLine(this IAnsiConsole console, IFormatProvider provider, string format, params object[] args)
|
public static void MarkupLine(this IAnsiConsole console, IFormatProvider provider, string format, params object[] args)
|
||||||
{
|
{
|
||||||
Markup(console, provider, format, args);
|
Markup(console, provider, format + Environment.NewLine, args);
|
||||||
console.WriteLine();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,40 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="IAnsiConsole"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static partial class AnsiConsoleExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a new <see cref="Progress"/> instance for the console.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="console">The console.</param>
|
||||||
|
/// <returns>A <see cref="Progress"/> instance.</returns>
|
||||||
|
public static Progress Progress(this IAnsiConsole console)
|
||||||
|
{
|
||||||
|
if (console is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(console));
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Progress(console);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a new <see cref="Status"/> instance for the console.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="console">The console.</param>
|
||||||
|
/// <returns>A <see cref="Status"/> instance.</returns>
|
||||||
|
public static Status Status(this IAnsiConsole console)
|
||||||
|
{
|
||||||
|
if (console is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(console));
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Status(console);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Collections.Generic;
|
||||||
using Spectre.Console.Rendering;
|
using Spectre.Console.Rendering;
|
||||||
|
|
||||||
namespace Spectre.Console
|
namespace Spectre.Console
|
||||||
@@ -26,19 +26,26 @@ namespace Spectre.Console
|
|||||||
throw new ArgumentNullException(nameof(renderable));
|
throw new ArgumentNullException(nameof(renderable));
|
||||||
}
|
}
|
||||||
|
|
||||||
var options = new RenderContext(console.Encoding, console.Capabilities.LegacyConsole);
|
var context = new RenderContext(console.Encoding, console.Capabilities.LegacyConsole);
|
||||||
var segments = renderable.Render(options, console.Width).ToArray();
|
var renderables = console.Pipeline.Process(context, new[] { renderable });
|
||||||
segments = Segment.Merge(segments).ToArray();
|
|
||||||
|
|
||||||
foreach (var segment in segments)
|
Render(console, context, renderables);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Render(IAnsiConsole console, RenderContext options, IEnumerable<IRenderable> renderables)
|
||||||
|
{
|
||||||
|
if (renderables is null)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(segment.Text))
|
return;
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.Write(segment.Text, segment.Style);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var result = new List<Segment>();
|
||||||
|
foreach (var renderable in renderables)
|
||||||
|
{
|
||||||
|
result.AddRange(renderable.Render(options, console.Width));
|
||||||
|
}
|
||||||
|
|
||||||
|
console.Write(Segment.Merge(result));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,26 @@ namespace Spectre.Console
|
|||||||
return new Recorder(console);
|
return new Recorder(console);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Writes the specified string value to the console.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="console">The console to write to.</param>
|
||||||
|
/// <param name="segment">The segment to write.</param>
|
||||||
|
public static void Write(this IAnsiConsole console, Segment segment)
|
||||||
|
{
|
||||||
|
if (console is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(console));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (segment is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(segment));
|
||||||
|
}
|
||||||
|
|
||||||
|
console.Write(new[] { segment });
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Writes the specified string value to the console.
|
/// Writes the specified string value to the console.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -25,7 +45,7 @@ namespace Spectre.Console
|
|||||||
/// <param name="text">The text to write.</param>
|
/// <param name="text">The text to write.</param>
|
||||||
public static void Write(this IAnsiConsole console, string text)
|
public static void Write(this IAnsiConsole console, string text)
|
||||||
{
|
{
|
||||||
Write(console, text, Style.Plain);
|
Render(console, new Text(text, Style.Plain));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -36,17 +56,7 @@ namespace Spectre.Console
|
|||||||
/// <param name="style">The text style.</param>
|
/// <param name="style">The text style.</param>
|
||||||
public static void Write(this IAnsiConsole console, string text, Style style)
|
public static void Write(this IAnsiConsole console, string text, Style style)
|
||||||
{
|
{
|
||||||
if (console is null)
|
Render(console, new Text(text, style));
|
||||||
{
|
|
||||||
throw new ArgumentNullException(nameof(console));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (text is null)
|
|
||||||
{
|
|
||||||
throw new ArgumentNullException(nameof(text));
|
|
||||||
}
|
|
||||||
|
|
||||||
console.Write(new Segment(text, style));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -60,7 +70,7 @@ namespace Spectre.Console
|
|||||||
throw new ArgumentNullException(nameof(console));
|
throw new ArgumentNullException(nameof(console));
|
||||||
}
|
}
|
||||||
|
|
||||||
console.Write(Environment.NewLine, Style.Plain);
|
Render(console, new Text(Environment.NewLine, Style.Plain));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -91,8 +101,7 @@ namespace Spectre.Console
|
|||||||
throw new ArgumentNullException(nameof(text));
|
throw new ArgumentNullException(nameof(text));
|
||||||
}
|
}
|
||||||
|
|
||||||
console.Write(new Segment(text, style));
|
console.Write(text + Environment.NewLine, style);
|
||||||
console.WriteLine();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ namespace Spectre.Console
|
|||||||
}
|
}
|
||||||
|
|
||||||
alignment ??= panel.Header?.Alignment;
|
alignment ??= panel.Header?.Alignment;
|
||||||
return Header(panel, new PanelHeader(text, alignment));
|
return Header(panel, new PanelHeader(text, alignment));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -0,0 +1,54 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="PercentageColumn"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class PercentageColumnExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style for a non-complete task.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static PercentageColumn Style(this PercentageColumn column, Style style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(style));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.Style = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style for a completed task.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static PercentageColumn CompletedStyle(this PercentageColumn column, Style style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(style));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.CompletedStyle = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,76 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="ProgressBarColumn"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class ProgressBarColumnExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style of completed portions of the progress bar.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static ProgressBarColumn CompletedStyle(this ProgressBarColumn column, Style style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(style));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.CompletedStyle = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style of a finished progress bar.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static ProgressBarColumn FinishedStyle(this ProgressBarColumn column, Style style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(style));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.FinishedStyle = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style of remaining portions of the progress bar.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static ProgressBarColumn RemainingStyle(this ProgressBarColumn column, Style style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(style));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.RemainingStyle = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,79 @@
|
|||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="Progress"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class ProgressExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the columns to be used for an <see cref="Progress"/> instance.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="progress">The <see cref="Progress"/> instance.</param>
|
||||||
|
/// <param name="columns">The columns to use.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static Progress Columns(this Progress progress, ProgressColumn[] columns)
|
||||||
|
{
|
||||||
|
if (progress is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(progress));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (columns is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(columns));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!columns.Any())
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("At least one column must be specified.");
|
||||||
|
}
|
||||||
|
|
||||||
|
progress.Columns.Clear();
|
||||||
|
progress.Columns.AddRange(columns);
|
||||||
|
|
||||||
|
return progress;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets whether or not auto refresh is enabled.
|
||||||
|
/// If disabled, you will manually have to refresh the progress.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="progress">The <see cref="Progress"/> instance.</param>
|
||||||
|
/// <param name="enabled">Whether or not auto refresh is enabled.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static Progress AutoRefresh(this Progress progress, bool enabled)
|
||||||
|
{
|
||||||
|
if (progress is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(progress));
|
||||||
|
}
|
||||||
|
|
||||||
|
progress.AutoRefresh = enabled;
|
||||||
|
|
||||||
|
return progress;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets whether or not auto clear is enabled.
|
||||||
|
/// If enabled, the task tabled will be removed once
|
||||||
|
/// all tasks have completed.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="progress">The <see cref="Progress"/> instance.</param>
|
||||||
|
/// <param name="enabled">Whether or not auto clear is enabled.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static Progress AutoClear(this Progress progress, bool enabled)
|
||||||
|
{
|
||||||
|
if (progress is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(progress));
|
||||||
|
}
|
||||||
|
|
||||||
|
progress.AutoClear = enabled;
|
||||||
|
|
||||||
|
return progress;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="ProgressTask"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class ProgressTaskExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the task description.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="task">The task.</param>
|
||||||
|
/// <param name="description">The description.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static ProgressTask Description(this ProgressTask task, string description)
|
||||||
|
{
|
||||||
|
if (task is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(task));
|
||||||
|
}
|
||||||
|
|
||||||
|
task.Description = description;
|
||||||
|
return task;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the max value of the task.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="task">The task.</param>
|
||||||
|
/// <param name="value">The max value.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static ProgressTask MaxValue(this ProgressTask task, double value)
|
||||||
|
{
|
||||||
|
if (task is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(task));
|
||||||
|
}
|
||||||
|
|
||||||
|
task.MaxValue = value;
|
||||||
|
return task;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="RemainingTimeColumn"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class RemainingTimeColumnExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style of the remaining time text.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static RemainingTimeColumn Style(this RemainingTimeColumn column, Style style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(style));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.Style = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="SpinnerColumn"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class SpinnerColumnExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the style of the spinner.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="column">The column.</param>
|
||||||
|
/// <param name="style">The style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static SpinnerColumn Style(this SpinnerColumn column, Style? style)
|
||||||
|
{
|
||||||
|
if (column is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(column));
|
||||||
|
}
|
||||||
|
|
||||||
|
column.Style = style;
|
||||||
|
return column;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="StatusContext"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class StatusContextExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the status message.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="context">The status context.</param>
|
||||||
|
/// <param name="status">The status message.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static StatusContext Status(this StatusContext context, string status)
|
||||||
|
{
|
||||||
|
if (context is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(context));
|
||||||
|
}
|
||||||
|
|
||||||
|
context.Status = status;
|
||||||
|
return context;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the spinner.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="context">The status context.</param>
|
||||||
|
/// <param name="spinner">The spinner.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static StatusContext Spinner(this StatusContext context, Spinner spinner)
|
||||||
|
{
|
||||||
|
if (context is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(context));
|
||||||
|
}
|
||||||
|
|
||||||
|
context.Spinner = spinner;
|
||||||
|
return context;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the spinner style.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="context">The status context.</param>
|
||||||
|
/// <param name="style">The spinner style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static StatusContext SpinnerStyle(this StatusContext context, Style? style)
|
||||||
|
{
|
||||||
|
if (context is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(context));
|
||||||
|
}
|
||||||
|
|
||||||
|
context.SpinnerStyle = style;
|
||||||
|
return context;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
62
src/Spectre.Console/Extensions/Progress/StatusExtensions.cs
Normal file
62
src/Spectre.Console/Extensions/Progress/StatusExtensions.cs
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Contains extension methods for <see cref="Status"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static class StatusExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sets whether or not auto refresh is enabled.
|
||||||
|
/// If disabled, you will manually have to refresh the progress.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="status">The <see cref="Status"/> instance.</param>
|
||||||
|
/// <param name="enabled">Whether or not auto refresh is enabled.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static Status AutoRefresh(this Status status, bool enabled)
|
||||||
|
{
|
||||||
|
if (status is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(status));
|
||||||
|
}
|
||||||
|
|
||||||
|
status.AutoRefresh = enabled;
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the spinner.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="status">The <see cref="Status"/> instance.</param>
|
||||||
|
/// <param name="spinner">The spinner.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static Status Spinner(this Status status, Spinner spinner)
|
||||||
|
{
|
||||||
|
if (status is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(status));
|
||||||
|
}
|
||||||
|
|
||||||
|
status.Spinner = spinner;
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the spinner style.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="status">The <see cref="Status"/> instance.</param>
|
||||||
|
/// <param name="style">The spinner style.</param>
|
||||||
|
/// <returns>The same instance so that multiple calls can be chained.</returns>
|
||||||
|
public static Status SpinnerStyle(this Status status, Style? style)
|
||||||
|
{
|
||||||
|
if (status is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(status));
|
||||||
|
}
|
||||||
|
|
||||||
|
status.SpinnerStyle = style;
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
29
src/Spectre.Console/Extensions/StringBuilderExtensions.cs
Normal file
29
src/Spectre.Console/Extensions/StringBuilderExtensions.cs
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
using System.Globalization;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
namespace Spectre.Console.Internal
|
||||||
|
{
|
||||||
|
internal static class StringBuilderExtensions
|
||||||
|
{
|
||||||
|
public static StringBuilder AppendWithStyle(this StringBuilder builder, Style? style, int? value)
|
||||||
|
{
|
||||||
|
return AppendWithStyle(builder, style, value?.ToString(CultureInfo.InvariantCulture));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static StringBuilder AppendWithStyle(this StringBuilder builder, Style? style, string? value)
|
||||||
|
{
|
||||||
|
value ??= string.Empty;
|
||||||
|
|
||||||
|
if (style != null)
|
||||||
|
{
|
||||||
|
return builder.Append('[')
|
||||||
|
.Append(style.ToMarkup())
|
||||||
|
.Append(']')
|
||||||
|
.Append(value.EscapeMarkup())
|
||||||
|
.Append("[/]");
|
||||||
|
}
|
||||||
|
|
||||||
|
return builder.Append(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -62,10 +62,15 @@ namespace Spectre.Console
|
|||||||
return text;
|
return text;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static string NormalizeLineEndings(this string? text, bool native = false)
|
internal static string? RemoveNewLines(this string? text)
|
||||||
|
{
|
||||||
|
return text?.ReplaceExact("\r\n", string.Empty)
|
||||||
|
?.ReplaceExact("\n", string.Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
internal static string NormalizeNewLines(this string? text, bool native = false)
|
||||||
{
|
{
|
||||||
text = text?.ReplaceExact("\r\n", "\n");
|
text = text?.ReplaceExact("\r\n", "\n");
|
||||||
text = text?.ReplaceExact("\r", string.Empty);
|
|
||||||
text ??= string.Empty;
|
text ??= string.Empty;
|
||||||
|
|
||||||
if (native && !_alreadyNormalized)
|
if (native && !_alreadyNormalized)
|
||||||
@@ -78,7 +83,7 @@ namespace Spectre.Console
|
|||||||
|
|
||||||
internal static string[] SplitLines(this string text)
|
internal static string[] SplitLines(this string text)
|
||||||
{
|
{
|
||||||
var result = text?.NormalizeLineEndings()?.Split(new[] { '\n' }, StringSplitOptions.None);
|
var result = text?.NormalizeNewLines()?.Split(new[] { '\n' }, StringSplitOptions.None);
|
||||||
return result ?? Array.Empty<string>();
|
return result ?? Array.Empty<string>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using Spectre.Console.Rendering;
|
using Spectre.Console.Rendering;
|
||||||
|
|
||||||
@@ -28,6 +29,11 @@ namespace Spectre.Console
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
IAnsiConsoleInput Input { get; }
|
IAnsiConsoleInput Input { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the render pipeline.
|
||||||
|
/// </summary>
|
||||||
|
RenderPipeline Pipeline { get; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the buffer width of the console.
|
/// Gets the buffer width of the console.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -45,9 +51,9 @@ namespace Spectre.Console
|
|||||||
void Clear(bool home);
|
void Clear(bool home);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Writes a string followed by a line terminator to the console.
|
/// Writes multiple segments to the console.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="segment">The segment to write.</param>
|
/// <param name="segments">The segments to write.</param>
|
||||||
void Write(Segment segment);
|
void Write(IEnumerable<Segment> segments);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
24
src/Spectre.Console/InteractionSupport.cs
Normal file
24
src/Spectre.Console/InteractionSupport.cs
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Determines interactivity support.
|
||||||
|
/// </summary>
|
||||||
|
public enum InteractionSupport
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interaction support should be
|
||||||
|
/// detected by the system.
|
||||||
|
/// </summary>
|
||||||
|
Detect = 0,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interactivity is supported.
|
||||||
|
/// </summary>
|
||||||
|
Yes = 1,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interactivity is not supported.
|
||||||
|
/// </summary>
|
||||||
|
No = 2,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -121,6 +121,8 @@ namespace Spectre.Console.Internal
|
|||||||
// Enabling failed.
|
// Enabling failed.
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isLegacy = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using Spectre.Console.Rendering;
|
using Spectre.Console.Rendering;
|
||||||
@@ -11,9 +12,11 @@ namespace Spectre.Console.Internal
|
|||||||
private readonly AnsiBuilder _ansiBuilder;
|
private readonly AnsiBuilder _ansiBuilder;
|
||||||
private readonly AnsiCursor _cursor;
|
private readonly AnsiCursor _cursor;
|
||||||
private readonly ConsoleInput _input;
|
private readonly ConsoleInput _input;
|
||||||
|
private readonly object _lock;
|
||||||
|
|
||||||
public Capabilities Capabilities { get; }
|
public Capabilities Capabilities { get; }
|
||||||
public Encoding Encoding { get; }
|
public Encoding Encoding { get; }
|
||||||
|
public RenderPipeline Pipeline { get; }
|
||||||
public IAnsiConsoleCursor Cursor => _cursor;
|
public IAnsiConsoleCursor Cursor => _cursor;
|
||||||
public IAnsiConsoleInput Input => _input;
|
public IAnsiConsoleInput Input => _input;
|
||||||
|
|
||||||
@@ -49,35 +52,59 @@ namespace Spectre.Console.Internal
|
|||||||
|
|
||||||
Capabilities = capabilities ?? throw new ArgumentNullException(nameof(capabilities));
|
Capabilities = capabilities ?? throw new ArgumentNullException(nameof(capabilities));
|
||||||
Encoding = _out.IsStandardOut() ? System.Console.OutputEncoding : Encoding.UTF8;
|
Encoding = _out.IsStandardOut() ? System.Console.OutputEncoding : Encoding.UTF8;
|
||||||
|
Pipeline = new RenderPipeline();
|
||||||
|
|
||||||
_ansiBuilder = new AnsiBuilder(Capabilities, linkHasher);
|
_ansiBuilder = new AnsiBuilder(Capabilities, linkHasher);
|
||||||
_cursor = new AnsiCursor(this);
|
_cursor = new AnsiCursor(this);
|
||||||
_input = new ConsoleInput();
|
_input = new ConsoleInput();
|
||||||
|
_lock = new object();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Clear(bool home)
|
public void Clear(bool home)
|
||||||
{
|
{
|
||||||
Write(Segment.Control("\u001b[2J"));
|
lock (_lock)
|
||||||
|
|
||||||
if (home)
|
|
||||||
{
|
{
|
||||||
Cursor.SetPosition(0, 0);
|
Write(new[] { Segment.Control("\u001b[2J") });
|
||||||
|
|
||||||
|
if (home)
|
||||||
|
{
|
||||||
|
Cursor.SetPosition(0, 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Write(Segment segment)
|
public void Write(IEnumerable<Segment> segments)
|
||||||
{
|
{
|
||||||
var parts = segment.Text.NormalizeLineEndings().Split(new[] { '\n' });
|
lock (_lock)
|
||||||
foreach (var (_, _, last, part) in parts.Enumerate())
|
|
||||||
{
|
{
|
||||||
if (!string.IsNullOrEmpty(part))
|
var builder = new StringBuilder();
|
||||||
|
foreach (var segment in segments)
|
||||||
{
|
{
|
||||||
_out.Write(_ansiBuilder.GetAnsi(part, segment.Style));
|
if (segment.IsControlCode)
|
||||||
|
{
|
||||||
|
builder.Append(segment.Text);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var parts = segment.Text.NormalizeNewLines().Split(new[] { '\n' });
|
||||||
|
foreach (var (_, _, last, part) in parts.Enumerate())
|
||||||
|
{
|
||||||
|
if (!string.IsNullOrEmpty(part))
|
||||||
|
{
|
||||||
|
builder.Append(_ansiBuilder.GetAnsi(part, segment.Style));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!last)
|
||||||
|
{
|
||||||
|
builder.Append(Environment.NewLine);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!last)
|
if (builder.Length > 0)
|
||||||
{
|
{
|
||||||
_out.Write(Environment.NewLine);
|
_out.Write(builder.ToString());
|
||||||
|
_out.Flush();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,12 +50,18 @@ namespace Spectre.Console.Internal
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var supportsInteraction = settings.Interactive == InteractionSupport.Yes;
|
||||||
|
if (settings.Interactive == InteractionSupport.Detect)
|
||||||
|
{
|
||||||
|
supportsInteraction = InteractivityDetector.IsInteractive();
|
||||||
|
}
|
||||||
|
|
||||||
var colorSystem = settings.ColorSystem == ColorSystemSupport.Detect
|
var colorSystem = settings.ColorSystem == ColorSystemSupport.Detect
|
||||||
? ColorSystemDetector.Detect(supportsAnsi)
|
? ColorSystemDetector.Detect(supportsAnsi)
|
||||||
: (ColorSystem)settings.ColorSystem;
|
: (ColorSystem)settings.ColorSystem;
|
||||||
|
|
||||||
// Get the capabilities
|
// Get the capabilities
|
||||||
var capabilities = new Capabilities(supportsAnsi, colorSystem, legacyConsole);
|
var capabilities = new Capabilities(supportsAnsi, colorSystem, legacyConsole, supportsInteraction);
|
||||||
|
|
||||||
// Create the renderer
|
// Create the renderer
|
||||||
if (supportsAnsi)
|
if (supportsAnsi)
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using Spectre.Console.Rendering;
|
using Spectre.Console.Rendering;
|
||||||
@@ -14,6 +15,7 @@ namespace Spectre.Console.Internal
|
|||||||
|
|
||||||
public Capabilities Capabilities { get; }
|
public Capabilities Capabilities { get; }
|
||||||
public Encoding Encoding { get; }
|
public Encoding Encoding { get; }
|
||||||
|
public RenderPipeline Pipeline { get; }
|
||||||
public IAnsiConsoleCursor Cursor => _cursor;
|
public IAnsiConsoleCursor Cursor => _cursor;
|
||||||
public IAnsiConsoleInput Input => _input;
|
public IAnsiConsoleInput Input => _input;
|
||||||
|
|
||||||
@@ -43,8 +45,9 @@ namespace Spectre.Console.Internal
|
|||||||
System.Console.SetOut(@out ?? throw new ArgumentNullException(nameof(@out)));
|
System.Console.SetOut(@out ?? throw new ArgumentNullException(nameof(@out)));
|
||||||
}
|
}
|
||||||
|
|
||||||
Encoding = System.Console.OutputEncoding;
|
|
||||||
Capabilities = capabilities;
|
Capabilities = capabilities;
|
||||||
|
Encoding = System.Console.OutputEncoding;
|
||||||
|
Pipeline = new RenderPipeline();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Clear(bool home)
|
public void Clear(bool home)
|
||||||
@@ -60,14 +63,22 @@ namespace Spectre.Console.Internal
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Write(Segment segment)
|
public void Write(IEnumerable<Segment> segments)
|
||||||
{
|
{
|
||||||
if (_lastStyle?.Equals(segment.Style) != true)
|
foreach (var segment in segments)
|
||||||
{
|
{
|
||||||
SetStyle(segment.Style);
|
if (segment.IsControlCode)
|
||||||
}
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
System.Console.Write(segment.Text.NormalizeLineEndings(native: true));
|
if (_lastStyle?.Equals(segment.Style) != true)
|
||||||
|
{
|
||||||
|
SetStyle(segment.Style);
|
||||||
|
}
|
||||||
|
|
||||||
|
System.Console.Write(segment.Text.NormalizeNewLines(native: true));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SetStyle(Style style)
|
private void SetStyle(Style style)
|
||||||
|
|||||||
@@ -41,12 +41,15 @@ namespace Spectre.Console.Internal
|
|||||||
{
|
{
|
||||||
var shortenTypes = (settings.Format & ExceptionFormats.ShortenTypes) != 0;
|
var shortenTypes = (settings.Format & ExceptionFormats.ShortenTypes) != 0;
|
||||||
var type = Emphasize(ex.Type, new[] { '.' }, settings.Style.Exception, shortenTypes, settings);
|
var type = Emphasize(ex.Type, new[] { '.' }, settings.Style.Exception, shortenTypes, settings);
|
||||||
|
|
||||||
var message = $"[{settings.Style.Message.ToMarkup()}]{ex.Message.EscapeMarkup()}[/]";
|
var message = $"[{settings.Style.Message.ToMarkup()}]{ex.Message.EscapeMarkup()}[/]";
|
||||||
return new Markup(string.Concat(type, ": ", message));
|
return new Markup(string.Concat(type, ": ", message));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Grid GetStackFrames(ExceptionInfo ex, ExceptionSettings settings)
|
private static Grid GetStackFrames(ExceptionInfo ex, ExceptionSettings settings)
|
||||||
{
|
{
|
||||||
|
var styles = settings.Style;
|
||||||
|
|
||||||
var grid = new Grid();
|
var grid = new Grid();
|
||||||
grid.AddColumn(new GridColumn().PadLeft(2).PadRight(0).NoWrap());
|
grid.AddColumn(new GridColumn().PadLeft(2).PadRight(0).NoWrap());
|
||||||
grid.AddColumn(new GridColumn().PadLeft(1).PadRight(0));
|
grid.AddColumn(new GridColumn().PadLeft(1).PadRight(0));
|
||||||
@@ -66,14 +69,16 @@ namespace Spectre.Console.Internal
|
|||||||
|
|
||||||
// Method
|
// Method
|
||||||
var shortenMethods = (settings.Format & ExceptionFormats.ShortenMethods) != 0;
|
var shortenMethods = (settings.Format & ExceptionFormats.ShortenMethods) != 0;
|
||||||
builder.Append(Emphasize(frame.Method, new[] { '.' }, settings.Style.Method, shortenMethods, settings));
|
builder.Append(Emphasize(frame.Method, new[] { '.' }, styles.Method, shortenMethods, settings));
|
||||||
builder.Append('[').Append(settings.Style.Parenthesis.ToMarkup()).Append(']').Append('(').Append("[/]");
|
builder.AppendWithStyle(styles.Parenthesis, "(");
|
||||||
AppendParameters(builder, frame, settings);
|
AppendParameters(builder, frame, settings);
|
||||||
builder.Append('[').Append(settings.Style.Parenthesis.ToMarkup()).Append(']').Append(')').Append("[/]");
|
builder.AppendWithStyle(styles.Parenthesis, ")");
|
||||||
|
|
||||||
if (frame.Path != null)
|
if (frame.Path != null)
|
||||||
{
|
{
|
||||||
builder.Append(" [").Append(settings.Style.Dimmed.ToMarkup()).Append("]in[/] ");
|
builder.Append(' ');
|
||||||
|
builder.AppendWithStyle(styles.Dimmed, "in");
|
||||||
|
builder.Append(' ');
|
||||||
|
|
||||||
// Path
|
// Path
|
||||||
AppendPath(builder, frame, settings);
|
AppendPath(builder, frame, settings);
|
||||||
@@ -81,13 +86,13 @@ namespace Spectre.Console.Internal
|
|||||||
// Line number
|
// Line number
|
||||||
if (frame.LineNumber != null)
|
if (frame.LineNumber != null)
|
||||||
{
|
{
|
||||||
builder.Append('[').Append(settings.Style.Dimmed.ToMarkup()).Append("]:[/]");
|
builder.AppendWithStyle(styles.Dimmed, ":");
|
||||||
builder.Append('[').Append(settings.Style.LineNumber.ToMarkup()).Append(']').Append(frame.LineNumber).Append("[/]");
|
builder.AppendWithStyle(styles.LineNumber, frame.LineNumber);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
grid.AddRow(
|
grid.AddRow(
|
||||||
$"[{settings.Style.Dimmed.ToMarkup()}]at[/]",
|
$"[{styles.Dimmed.ToMarkup()}]at[/]",
|
||||||
builder.ToString());
|
builder.ToString());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -98,7 +103,7 @@ namespace Spectre.Console.Internal
|
|||||||
{
|
{
|
||||||
var typeColor = settings.Style.ParameterType.ToMarkup();
|
var typeColor = settings.Style.ParameterType.ToMarkup();
|
||||||
var nameColor = settings.Style.ParameterName.ToMarkup();
|
var nameColor = settings.Style.ParameterName.ToMarkup();
|
||||||
var parameters = frame.Parameters.Select(x => $"[{typeColor}]{x.Type.EscapeMarkup()}[/] [{nameColor}]{x.Name}[/]");
|
var parameters = frame.Parameters.Select(x => $"[{typeColor}]{x.Type.EscapeMarkup()}[/] [{nameColor}]{x.Name.EscapeMarkup()}[/]");
|
||||||
builder.Append(string.Join(", ", parameters));
|
builder.Append(string.Join(", ", parameters));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -146,16 +151,18 @@ namespace Spectre.Console.Internal
|
|||||||
{
|
{
|
||||||
if (!compact)
|
if (!compact)
|
||||||
{
|
{
|
||||||
builder.Append('[').Append(settings.Style.NonEmphasized.ToMarkup()).Append(']')
|
builder.AppendWithStyle(
|
||||||
.Append(type, 0, index + 1).Append("[/]");
|
settings.Style.NonEmphasized,
|
||||||
|
type.Substring(0, index + 1).EscapeMarkup());
|
||||||
}
|
}
|
||||||
|
|
||||||
builder.Append('[').Append(color.ToMarkup()).Append(']')
|
builder.AppendWithStyle(
|
||||||
.Append(type, index + 1, type.Length - index - 1).Append("[/]");
|
color,
|
||||||
|
type.Substring(index + 1, type.Length - index - 1).EscapeMarkup());
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
builder.Append(type);
|
builder.Append(type.EscapeMarkup());
|
||||||
}
|
}
|
||||||
|
|
||||||
return builder.ToString();
|
return builder.ToString();
|
||||||
|
|||||||
@@ -15,6 +15,11 @@ namespace Spectre.Console.Internal
|
|||||||
|
|
||||||
foreach (var (_, first, _, segment) in segments.Enumerate())
|
foreach (var (_, first, _, segment) in segments.Enumerate())
|
||||||
{
|
{
|
||||||
|
if (segment.IsControlCode)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (segment.Text == "\n" && !first)
|
if (segment.Text == "\n" && !first)
|
||||||
{
|
{
|
||||||
builder.Append('\n');
|
builder.Append('\n');
|
||||||
|
|||||||
52
src/Spectre.Console/Internal/InteractivityDetector.cs
Normal file
52
src/Spectre.Console/Internal/InteractivityDetector.cs
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace Spectre.Console.Internal
|
||||||
|
{
|
||||||
|
internal static class InteractivityDetector
|
||||||
|
{
|
||||||
|
private static readonly Dictionary<string, Func<string, bool>> _environmentVariables;
|
||||||
|
|
||||||
|
static InteractivityDetector()
|
||||||
|
{
|
||||||
|
_environmentVariables = new Dictionary<string, Func<string, bool>>
|
||||||
|
{
|
||||||
|
{ "APPVEYOR", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "bamboo_buildNumber", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "BITBUCKET_REPO_OWNER", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "BITBUCKET_REPO_SLUG", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "BITBUCKET_COMMIT", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "BITRISE_BUILD_URL", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "ContinuaCI.Version", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "CI_SERVER", v => v.Equals("yes", StringComparison.OrdinalIgnoreCase) }, // GitLab
|
||||||
|
{ "GITHUB_ACTIONS", v => v.Equals("true", StringComparison.OrdinalIgnoreCase) },
|
||||||
|
{ "GO_SERVER_URL", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "JENKINS_URL", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "BuildRunner", v => v.Equals("MyGet", StringComparison.OrdinalIgnoreCase) },
|
||||||
|
{ "TEAMCITY_VERSION", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
{ "TF_BUILD", v => !string.IsNullOrWhiteSpace(v) }, // TFS and Azure
|
||||||
|
{ "TRAVIS", v => !string.IsNullOrWhiteSpace(v) },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool IsInteractive()
|
||||||
|
{
|
||||||
|
if (!Environment.UserInteractive)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var variable in _environmentVariables)
|
||||||
|
{
|
||||||
|
var func = variable.Value;
|
||||||
|
var value = Environment.GetEnvironmentVariable(variable.Key);
|
||||||
|
if (!string.IsNullOrWhiteSpace(value) && variable.Value(value))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -24,7 +24,7 @@ namespace Spectre.Console.Internal
|
|||||||
|
|
||||||
using (var reader = new StreamReader(stream))
|
using (var reader = new StreamReader(stream))
|
||||||
{
|
{
|
||||||
return reader.ReadToEnd().NormalizeLineEndings();
|
return reader.ReadToEnd().NormalizeNewLines();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,32 +8,34 @@ namespace Spectre.Console.Internal
|
|||||||
{
|
{
|
||||||
public static int GetCellLength(RenderContext context, string text)
|
public static int GetCellLength(RenderContext context, string text)
|
||||||
{
|
{
|
||||||
return text.Sum(rune =>
|
return text.Sum(rune => GetCellLength(context, rune));
|
||||||
{
|
}
|
||||||
if (context.LegacyConsole)
|
|
||||||
{
|
|
||||||
// Is it represented by a single byte?
|
|
||||||
// In that case we don't have to calculate the
|
|
||||||
// actual cell width.
|
|
||||||
if (context.Encoding.GetByteCount(new[] { rune }) == 1)
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: We need to figure out why Segment.SplitLines fails
|
public static int GetCellLength(RenderContext context, char rune)
|
||||||
// if we let wcwidth (which returns -1 instead of 1)
|
{
|
||||||
// calculate the size for new line characters.
|
if (context.LegacyConsole)
|
||||||
// That is correct from a Unicode perspective, but the
|
{
|
||||||
// algorithm was written before wcwidth was added and used
|
// Is it represented by a single byte?
|
||||||
// to work with string length and not cell length.
|
// In that case we don't have to calculate the
|
||||||
if (rune == '\n')
|
// actual cell width.
|
||||||
|
if (context.Encoding.GetByteCount(new[] { rune }) == 1)
|
||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return UnicodeCalculator.GetWidth(rune);
|
// TODO: We need to figure out why Segment.SplitLines fails
|
||||||
});
|
// if we let wcwidth (which returns -1 instead of 1)
|
||||||
|
// calculate the size for new line characters.
|
||||||
|
// That is correct from a Unicode perspective, but the
|
||||||
|
// algorithm was written before wcwidth was added and used
|
||||||
|
// to work with string length and not cell length.
|
||||||
|
if (rune == '\n')
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return UnicodeCalculator.GetWidth(rune);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,11 @@ namespace Spectre.Console.Internal
|
|||||||
|
|
||||||
foreach (var segment in Segment.Merge(segments))
|
foreach (var segment in Segment.Merge(segments))
|
||||||
{
|
{
|
||||||
|
if (segment.IsControlCode)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
builder.Append(segment.Text);
|
builder.Append(segment.Text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
35
src/Spectre.Console/Progress/Columns/PercentageColumn.cs
Normal file
35
src/Spectre.Console/Progress/Columns/PercentageColumn.cs
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
using System;
|
||||||
|
using Spectre.Console.Rendering;
|
||||||
|
|
||||||
|
namespace Spectre.Console
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// A column showing task progress in percentage.
|
||||||
|
/// </summary>
|
||||||
|
public sealed class PercentageColumn : ProgressColumn
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the style for a non-complete task.
|
||||||
|
/// </summary>
|
||||||
|
public Style Style { get; set; } = Style.Plain;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the style for a completed task.
|
||||||
|
/// </summary>
|
||||||
|
public Style CompletedStyle { get; set; } = new Style(foreground: Color.Green);
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public override IRenderable Render(RenderContext context, ProgressTask task, TimeSpan deltaTime)
|
||||||
|
{
|
||||||
|
var percentage = (int)task.Percentage;
|
||||||
|
var style = percentage == 100 ? CompletedStyle : Style ?? Style.Plain;
|
||||||
|
return new Text($"{percentage}%", style).RightAligned();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public override int? GetColumnWidth(RenderContext context)
|
||||||
|
{
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user