2019-08-07 22:25:33 +02:00
using System ;
2019-08-11 18:02:21 +02:00
using System.Collections.Generic ;
2019-08-18 18:32:34 +02:00
using System.Linq ;
2019-08-11 18:02:21 +02:00
using Coroutine ;
2019-08-07 22:25:33 +02:00
using Microsoft.Xna.Framework ;
2019-08-09 15:15:22 +02:00
using Microsoft.Xna.Framework.Graphics ;
2019-08-16 19:08:36 +02:00
using MLEM.Extensions ;
2019-08-09 19:28:48 +02:00
using MLEM.Font ;
2019-08-07 22:25:33 +02:00
using MLEM.Input ;
2019-08-07 21:26:16 +02:00
using MLEM.Startup ;
2019-08-09 15:15:22 +02:00
using MLEM.Textures ;
2019-08-09 18:26:28 +02:00
using MLEM.Ui ;
using MLEM.Ui.Elements ;
2019-08-10 21:37:10 +02:00
using MLEM.Ui.Style ;
2019-08-07 21:26:16 +02:00
2019-08-14 14:15:07 +02:00
namespace Demos {
/// <remarks>
/// NOTE: This ui demo derives from <see cref="MlemGame"/>. To use MLEM.Ui, it's not required to use MLEM.Startup (which MlemGame is a part of).
/// If using your own game class that derives from <see cref="Game"/>, however, you will have to do a few additional things to get MLEM.Ui up and running:
/// - Create an instance of <see cref="UiSystem"/>
/// - Call the instance's Update method in your game's Update method
2019-08-15 14:59:15 +02:00
/// - Call the instance's DrawEarly method before clearing your <see cref="GraphicsDevice"/>
2019-08-14 14:15:07 +02:00
/// - Call the instance's Draw method in your game's Draw method
/// </remarks>
2019-09-01 11:55:41 +02:00
public class UiDemo : Demo {
2019-08-07 21:26:16 +02:00
2019-08-09 15:15:22 +02:00
private Texture2D testTexture ;
private NinePatch testPatch ;
2019-08-09 18:26:28 +02:00
2019-09-01 11:55:41 +02:00
public UiDemo ( MlemGame game ) : base ( game ) {
2019-08-09 22:04:26 +02:00
}
2019-09-01 11:55:41 +02:00
public override void LoadContent ( ) {
2019-08-09 15:15:22 +02:00
this . testTexture = LoadContent < Texture2D > ( "Textures/Test" ) ;
this . testPatch = new NinePatch ( new TextureRegion ( this . testTexture , 0 , 8 , 24 , 24 ) , 8 ) ;
2019-08-10 21:37:10 +02:00
base . LoadContent ( ) ;
2019-08-09 18:26:28 +02:00
2019-08-14 14:15:07 +02:00
// create a new style
// this one derives form UntexturedStyle so that stuff like the hover colors don't have to be set again
2019-08-12 19:44:16 +02:00
var style = new UntexturedStyle ( this . SpriteBatch ) {
2019-08-14 14:15:07 +02:00
// when using a SpriteFont, use GenericSpriteFont. When using a MonoGame.Extended BitmapFont, use GenericBitmapFont.
// Wrapping fonts like this allows for both types to be usable within MLEM.Ui easily
2019-08-10 21:37:10 +02:00
Font = new GenericSpriteFont ( LoadContent < SpriteFont > ( "Fonts/TestFont" ) ) ,
2019-08-24 00:07:54 +02:00
BoldFont = new GenericSpriteFont ( LoadContent < SpriteFont > ( "Fonts/TestFontBold" ) ) ,
ItalicFont = new GenericSpriteFont ( LoadContent < SpriteFont > ( "Fonts/TestFontItalic" ) ) ,
2019-08-13 21:23:20 +02:00
TextScale = 0.1F ,
2019-08-10 21:37:10 +02:00
PanelTexture = this . testPatch ,
ButtonTexture = new NinePatch ( new TextureRegion ( this . testTexture , 24 , 8 , 16 , 16 ) , 4 ) ,
TextFieldTexture = new NinePatch ( new TextureRegion ( this . testTexture , 24 , 8 , 16 , 16 ) , 4 ) ,
2019-08-13 23:54:29 +02:00
ScrollBarBackground = new NinePatch ( new TextureRegion ( this . testTexture , 12 , 0 , 4 , 8 ) , 1 , 1 , 2 , 2 ) ,
ScrollBarScrollerTexture = new NinePatch ( new TextureRegion ( this . testTexture , 8 , 0 , 4 , 8 ) , 1 , 1 , 2 , 2 ) ,
2019-08-13 21:23:20 +02:00
CheckboxTexture = new NinePatch ( new TextureRegion ( this . testTexture , 24 , 8 , 16 , 16 ) , 4 ) ,
CheckboxCheckmark = new TextureRegion ( this . testTexture , 24 , 0 , 8 , 8 ) ,
RadioTexture = new NinePatch ( new TextureRegion ( this . testTexture , 16 , 0 , 8 , 8 ) , 3 ) ,
RadioCheckmark = new TextureRegion ( this . testTexture , 32 , 0 , 8 , 8 )
2019-08-10 21:37:10 +02:00
} ;
2019-09-01 11:55:41 +02:00
var untexturedStyle = new UntexturedStyle ( this . SpriteBatch ) ;
2019-08-14 14:15:07 +02:00
// set the defined style as the current one
2019-08-10 21:37:10 +02:00
this . UiSystem . Style = style ;
2019-08-14 14:15:07 +02:00
// scale every ui up by 5
2019-08-11 21:24:09 +02:00
this . UiSystem . GlobalScale = 5 ;
2019-08-23 19:46:36 +02:00
// set the ui system to automatically scale with screen size
// this will cause all ui elements to be scaled based on the reference resolution (AutoScaleReferenceSize)
// by default, the reference resolution is set to the initial screen size, however this value can be changed through the ui system
this . UiSystem . AutoScaleWithScreen = true ;
2019-08-09 18:26:28 +02:00
2019-08-14 14:15:07 +02:00
// create the root panel that all the other components sit on and add it to the ui system
2019-08-13 23:54:29 +02:00
var root = new Panel ( Anchor . Center , new Vector2 ( 80 , 100 ) , Vector2 . Zero , false , true , new Point ( 5 , 10 ) ) ;
2019-08-10 21:37:10 +02:00
this . UiSystem . Add ( "Test" , root ) ;
2019-08-25 19:07:45 +02:00
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "This is a small demo for MLEM.Ui, a user interface library that is part of (M)LEM (L)ibrary by (E)llpeck for (M)onoGame." ) ) ;
2019-08-13 21:27:27 +02:00
var image = root . AddChild ( new Image ( Anchor . AutoCenter , new Vector2 ( 50 , 50 ) , new TextureRegion ( this . testTexture , 0 , 0 , 8 , 8 ) ) { IsHidden = true , Padding = new Point ( 3 ) } ) ;
2019-08-23 00:15:00 +02:00
// Setting the x or y coordinate of the size to 1 or a lower number causes the width or height to be a percentage of the parent's width or height
2019-08-18 17:59:14 +02:00
// (for example, setting the size's x to 0.75 would make the element's width be 0.75*parentWidth)
2019-08-13 23:54:29 +02:00
root . AddChild ( new Button ( Anchor . AutoCenter , new Vector2 ( 1 , 10 ) , "Toggle Test Image" , "This button shows a grass tile as a test image to show the automatic anchoring of objects." ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > image . IsHidden = ! image . IsHidden
2019-08-09 22:04:26 +02:00
} ) ;
2019-08-24 12:40:20 +02:00
2019-08-13 21:27:27 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "Note that the default style does not contain any textures or font files and, as such, is quite bland. However, the default style is quite easy to override." ) ) ;
2019-08-13 21:23:20 +02:00
root . AddChild ( new Button ( Anchor . AutoCenter , new Vector2 ( 1 , 10 ) , "Change Style" ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > this . UiSystem . Style = this . UiSystem . Style = = untexturedStyle ? style : untexturedStyle ,
2019-08-13 23:54:29 +02:00
PositionOffset = new Vector2 ( 0 , 1 ) ,
2019-08-14 14:15:07 +02:00
// set HasCustomStyle to true before changing style information so that, when changing the style globally
// (like above), these custom values don't get undone
2019-08-10 21:37:10 +02:00
HasCustomStyle = true ,
Texture = this . testPatch ,
2019-08-28 18:27:17 +02:00
HoveredColor = Color . LightGray ,
SelectionIndicator = style . SelectionIndicator
2019-08-10 21:37:10 +02:00
} ) ;
2019-08-11 18:02:21 +02:00
2019-08-24 00:07:54 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
// a paragraph with formatting codes. To see them all or to add more, check the TextFormatting class
2019-08-24 12:40:20 +02:00
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "Paragraphs can also contain [Blue]formatting codes[White], including colors and [Italic]text styles[Regular]. The names of all [Orange]MonoGame Colors[White] can be used, as well as the codes [Italic]Italic[Regular] and [Bold]Bold[Regular]. \n[Italic]Even [CornflowerBlue]Cornflower Blue[White] works!" ) ) ;
2019-08-24 00:07:54 +02:00
2019-08-11 18:02:21 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
2019-08-13 21:23:20 +02:00
root . AddChild ( new Paragraph ( Anchor . AutoCenter , 1 , "Text input:" , true ) ) ;
2019-08-23 18:56:39 +02:00
root . AddChild ( new TextField ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) ) {
PositionOffset = new Vector2 ( 0 , 1 ) ,
PlaceholderText = "Click here to input text"
} ) ;
2019-08-13 21:23:20 +02:00
2019-08-18 17:59:14 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new Paragraph ( Anchor . AutoCenter , 1 , "Numbers-only input:" , true ) ) ;
root . AddChild ( new TextField ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , TextField . OnlyNumbers ) { PositionOffset = new Vector2 ( 0 , 1 ) } ) ;
2019-08-13 21:23:20 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "Zoom in or out:" ) ) ;
root . AddChild ( new Button ( Anchor . AutoLeft , new Vector2 ( 10 ) , "+" ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > {
2019-08-11 18:02:21 +02:00
if ( element . Root . Scale < 2 )
element . Root . Scale + = 0.1F ;
}
} ) ;
2019-08-13 21:23:20 +02:00
root . AddChild ( new Button ( Anchor . AutoInline , new Vector2 ( 10 ) , "-" ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > {
2019-08-11 18:02:21 +02:00
if ( element . Root . Scale > 0.5F )
element . Root . Scale - = 0.1F ;
2019-08-13 21:23:20 +02:00
} ,
2019-08-13 23:54:29 +02:00
PositionOffset = new Vector2 ( 1 , 0 )
2019-08-11 21:24:09 +02:00
} ) ;
2019-08-13 21:23:20 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new Checkbox ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , "Checkbox 1!" ) ) ;
2019-08-13 23:54:29 +02:00
root . AddChild ( new Checkbox ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , "Checkbox 2!" ) { PositionOffset = new Vector2 ( 0 , 1 ) } ) ;
2019-08-13 21:23:20 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new RadioButton ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , "Radio button 1!" ) ) ;
2019-08-13 23:54:29 +02:00
root . AddChild ( new RadioButton ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , "Radio button 2!" ) { PositionOffset = new Vector2 ( 0 , 1 ) } ) ;
2019-08-15 14:59:15 +02:00
var tooltip = new Tooltip ( 50 , "This is a test tooltip to see the window bounding" ) { IsHidden = true } ;
2019-08-28 18:27:17 +02:00
this . UiSystem . Add ( "TestTooltip" , tooltip ) . CanSelectContent = false ;
2019-08-15 14:59:15 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new Button ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , "Toggle Test Tooltip" ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > tooltip . IsHidden = ! tooltip . IsHidden
2019-08-15 14:59:15 +02:00
} ) ;
2019-08-16 19:08:36 +02:00
var slider = new Slider ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , 5 , 1 ) ;
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , paragraph = > "Slider is at " + ( slider . CurrentValue * 100 ) . Floor ( ) + "%" ) { PositionOffset = new Vector2 ( 0 , 1 ) } ) ;
root . AddChild ( slider ) ;
2019-08-18 18:32:34 +02:00
2019-08-20 21:35:53 +02:00
// This button uses a coroutine from my Coroutine NuGet package (which is included with MLEM.Startup)
// but the important thing it does is change its visual scale and offset (check the method below for more info)
root . AddChild ( new Button ( Anchor . AutoCenter , new Vector2 ( 0.5F , 10 ) , "Wobble" , "This button wobbles around visually when clicked, but this doesn't affect its actual size and positioning" ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > CoroutineHandler . Start ( this . WobbleButton ( element ) ) ,
2019-08-20 21:35:53 +02:00
PositionOffset = new Vector2 ( 0 , 1 )
} ) ;
2019-08-24 12:40:20 +02:00
root . AddChild ( new VerticalSpace ( 3 ) ) ;
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "There are also some additional \"components\" which are created as combinations of other components. You can find all of them in the ElementHelper class. Here are some examples:" ) ) ;
root . AddChild ( ElementHelper . NumberField ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) ) ) . PositionOffset = new Vector2 ( 0 , 1 ) ;
2019-08-24 14:34:08 +02:00
root . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "There is an easy helper method to make any amount of same-sized columns:" ) { PositionOffset = new Vector2 ( 0 , 1 ) } ) ;
2019-08-24 15:12:11 +02:00
var cols = ElementHelper . MakeColumns ( root , new Vector2 ( 1 ) , 3 ) ;
2019-08-24 14:34:08 +02:00
cols [ 0 ] . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "This is the first column" ) ) ;
cols [ 1 ] . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "This is the second column" ) ) ;
cols [ 2 ] . AddChild ( new Paragraph ( Anchor . AutoLeft , 1 , "This is the third column" ) ) ;
2019-08-24 15:00:08 +02:00
root . AddChild ( new Button ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , "Show Info Box" ) {
2019-08-25 21:49:27 +02:00
OnPressed = element = > ElementHelper . ShowInfoBox ( this . UiSystem , Anchor . Center , 100 , "This is an easy info box that you can open with just one line of code! It automatically closes when you press the button below as well." ) ,
2019-08-24 15:00:08 +02:00
PositionOffset = new Vector2 ( 0 , 1 )
} ) ;
2019-08-27 21:44:02 +02:00
var region = new TextureRegion ( LoadContent < Texture2D > ( "Textures/Tree" ) ) ;
root . AddChild ( ElementHelper . ImageButton ( Anchor . AutoLeft , new Vector2 ( 1 , 10 ) , region , "Button with image" ) ) . PositionOffset = new Vector2 ( 0 , 1 ) ;
2019-08-18 18:32:34 +02:00
// Below are some querying examples that help you find certain elements easily
2019-08-18 18:34:15 +02:00
var children = root . GetChildren ( ) ;
var totalChildren = root . GetChildren ( regardChildrensChildren : true ) ;
Console . WriteLine ( $"The root has {children.Count()} children, but there are {totalChildren.Count()} when regarding children's children" ) ;
2019-08-20 21:35:53 +02:00
2019-08-18 18:32:34 +02:00
var textFields = root . GetChildren < TextField > ( ) ;
Console . WriteLine ( $"The root has {textFields.Count()} text fields" ) ;
var paragraphs = root . GetChildren < Paragraph > ( ) ;
var totalParagraphs = root . GetChildren < Paragraph > ( regardChildrensChildren : true ) ;
Console . WriteLine ( $"The root has {paragraphs.Count()} paragraphs, but there are {totalParagraphs.Count()} when regarding children's children" ) ;
var autoWidthChildren = root . GetChildren ( e = > e . Size . X = = 1 ) ;
var autoWidthButtons = root . GetChildren < Button > ( e = > e . Size . X = = 1 ) ;
Console . WriteLine ( $"The root has {autoWidthChildren.Count()} auto-width children, {autoWidthButtons.Count()} of which are buttons" ) ;
2019-08-07 22:25:33 +02:00
}
2019-08-07 21:26:16 +02:00
2019-08-20 21:35:53 +02:00
// This method is used by the wobbling button (see above)
// Note that this particular example makes use of the Coroutine package
private IEnumerator < Wait > WobbleButton ( Element button ) {
var counter = 0F ;
2019-08-24 19:52:28 +02:00
while ( counter < 4 * Math . PI ) {
2019-08-20 21:35:53 +02:00
// The imporant bit is that it changes its added display scale and offset, allowing the button to still maintain the
// correct position and scaling for both anchoring and interacting purposes, but to show any kind of animation visually
// This could be useful, for example, to create a little feedback effect to clicking it where it changes size for a second
button . AddedDisplayScale = new Vector2 ( ( float ) Math . Sin ( counter ) ) ;
button . AddedDisplayOffset = new Vector2 ( ( float ) Math . Sin ( counter / 2 ) * 4 , 0 ) ;
counter + = 0.1F ;
yield return new WaitSeconds ( 0.01F ) ;
}
button . AddedDisplayScale = Vector2 . Zero ;
button . AddedDisplayOffset = Vector2 . Zero ;
}
2019-09-01 11:55:41 +02:00
public override void Clear ( ) {
this . UiSystem . Remove ( "Test" ) ;
this . UiSystem . Remove ( "TestTooltip" ) ;
}
public override void DoDraw ( GameTime gameTime ) {
2019-08-15 14:59:15 +02:00
this . GraphicsDevice . Clear ( Color . CornflowerBlue ) ;
base . DoDraw ( gameTime ) ;
2019-08-09 15:15:22 +02:00
}
2019-08-07 21:26:16 +02:00
}
}