diff --git a/.gitignore b/.gitignore index e69de29..a48cf0d 100644 --- a/.gitignore +++ b/.gitignore @@ -0,0 +1 @@ +public diff --git a/.travis.yml b/.travis.yml index e3a70df..4e3c728 100644 --- a/.travis.yml +++ b/.travis.yml @@ -20,7 +20,7 @@ deploy: # The output of our static site local_dir: public # The repository we are deploying to - repo: encompass-ecs/encompass-ecs.github.io + repo: MoonsideGames/moonsidegames.github.io # The branch we are pushing the static repository target_branch: master # Information to use in the commit diff --git a/config.toml b/config.toml index 2a886a9..bd44d9c 100644 --- a/config.toml +++ b/config.toml @@ -1,4 +1,5 @@ baseURL = "/" +publishDir = "public/docs/encompass" languageCode = "en-US" title = "Encompass Docs" theme = ["hugo-notice", "hugo-theme-learn"] diff --git a/content/_index.md b/content/_index.md index 5dc6d44..8f9751a 100644 --- a/content/_index.md +++ b/content/_index.md @@ -6,13 +6,8 @@ title: "Encompass" [Encompass](https://github.com/encompass-ecs) is a powerful engine-agnostic framework to help you code games, or other kinds of simulations. -Object-oriented code is messy and rapidly becomes unmaintainable. +Object-oriented code scales and iterates poorly and rapidly becomes unmaintainable. -Encompass lets you write clean, de-coupled code so you can spend more time on your game design and less time fixing bugs. - -Encompass is currently available with a TypeScript implementation that fully supports transpilation to Javascript and -[Lua](https://github.com/TypeScriptToLua/TypeScriptToLua). - -A C# implementation is forthcoming. +Encompass lets you write clean, de-coupled code so you can spend more time on your game design and less time worrying about inheritance chains and fixing bugs. If you are here to learn how to use Encompass and don't care about the justifications for it, or you've never made a game before, I recommend skipping ahead to [Chapter 2](getting_started). diff --git a/content/concepts/component.md b/content/concepts/component.md index 4cb621e..3ba39a1 100644 --- a/content/concepts/component.md +++ b/content/concepts/component.md @@ -6,28 +6,33 @@ weight: 5 A Component is a collection of related data. -To define a Component, extend the Component class. +To define a Component, declare a struct which implements the **IComponent** interface. -```ts -import { Component } from "encompass-ecs"; +```cs +using Encompass; +using System.Numerics; -class PositionComponent extends Component { - public x: number; - public y: number; +public struct VelocityComponent : IComponent { + public Vector2 velocity; } ``` -Components are created in context with an Entity. +Components are attached to Entities with the **SetComponent** method. -```ts -const entity = World.create_entity(); -const position = entity.add_component(PositionComponent); -position.x = 3; -position.y = -4; +```cs +using Encompass; + +... + +var worldBuilder = new WorldBuilder(); +var entity = worldBuilder.CreateEntity(); +worldBuilder.SetComponent(entity, new VelocityComponent { velocity = Vector2.Zero }); ``` +**SetComponent** can also be used from within an **Engine**. We will talk more about this later. + Components cannot exist apart from an Entity and are automagically destroyed when they are removed or their Entity is destroyed. {{% notice warning %}} -Components should **never** reference other Components. This breaks the principle of loose coupling. You **will** regret it if you do this. +Components should **never** reference other Components directly. This breaks the principle of loose coupling. You **will** regret it if you do this. {{% /notice %}} diff --git a/content/concepts/engine.md b/content/concepts/engine.md index 62a5365..44d397f 100644 --- a/content/concepts/engine.md +++ b/content/concepts/engine.md @@ -10,29 +10,51 @@ An Engine is the Encompass notion of an ECS System. Much like the engine on a tr I never liked the term System. It is typically used to mean structures in game design and I found this confusing when discussing code implementation vs design. {{% /notice %}} -Engines are responsible for reading the game state, reading messages, emitting messages, and creating or mutating Entities and Components. +Engines may read any Entities and Components in the game world, read and send messages, and create and update Entities and Components. -An Engine which Reads a particular message is guaranteed to run *after* all Engines which Emit that particular message. +An Engine which Reads a particular message is guaranteed to run *after* all Engines which Send that particular message. To define an Engine, extend the Engine class. -Here is an example Engine: +Let's say we wanted to allow Entities to temporarily pause their motion for a specified amount of time. Here is an example Engine: -```ts -import { Engine, Mutates, Reads } from "encompass-ecs"; -import { LogoUIComponent } from "../../components/ui/logo"; -import { ShowUIMessage } from "../../messages/show_ui"; +```cs +using Encompass; -@Reads(ShowUIMessage) -@Mutates(LogoUIComponent) -export class LogoDisplayEngine extends Engine { - public update() { - const logo_ui_component = this.read_component_mutable(LogoUIComponent); - if (logo_ui_component && this.some(ShowUIMessage)) { - logo_ui_component.image.isVisible = true; +[Reads(typeof(PauseComponent))] +[Receives(typeof(PauseMessage))] +[Writes(typeof(PauseComponent))] +public class PauseEngine : Engine +{ + public override void Update(double dt) + { + foreach (var (pauseComponent, entity) in ReadComponentsIncludingEntity()) + { + var timer = pauseComponent.timer; + timer -= dt; + + if (timer <= 0) + { + RemoveComponent(entity); + } + else + { + SetComponent(entity, new PauseComponent { timer = timer }); + } + } + + foreach (var pauseMessage in ReadMessages()) + { + SetComponent(pauseMessage.entity, new PauseComponent + { + timer = pauseMessage.time + }); + } } - } } + ``` -If a LogoUIComponent exists, and a ShowUIMessage is received, it will make the logo image on the LogoUIComponent visible. Simple! +This engine deals with a Component called a PauseComponent. PauseComponent has a timer which counts down based on delta-time. When the timer ticks past zero, the PauseComponent is removed. If a PauseMessage is received, a new PauseComponent is attached to the Entity specified by the message. + +Notice that this Engine doesn't actually "do" the pausing, or even care if the Entity in question is capable of movement or not. In Engines that deals with movement, we can check if the Entities being moved have PauseComponents and modify how they are updated accordingly. This is the power of de-coupled logic at work. diff --git a/content/concepts/entity.md b/content/concepts/entity.md index 14d0db1..a50127e 100644 --- a/content/concepts/entity.md +++ b/content/concepts/entity.md @@ -8,12 +8,8 @@ An Entity is a structure composed of a unique ID and a collection of Components. Entities do not have any implicit properties or behaviors. They are granted these by their collection of Components. -There is no limit to the amount of Components an Entity may have, and Entities can have any number of Components of a particular type. +There is no limit to the amount of Components an Entity may have, but Entities may only have a single Component of a particular type. -Entities are active by default and can be deactivated. They can also be destroyed, permanently removing them and their components from the World. +Entities can also be destroyed, permanently removing them and their components from the World. Entities are created either by the WorldBuilder or by Engines. (We'll get into these soon.) - -{{% notice warning %}} -You should **never** add methods or properties to an Entity. This is what Components are for. -{{% /notice %}} diff --git a/content/concepts/message.md b/content/concepts/message.md index 60f52db..7641bca 100644 --- a/content/concepts/message.md +++ b/content/concepts/message.md @@ -8,19 +8,18 @@ Similar to Components, Messages are collections of data. Messages are used to transmit data between Engines so they can manipulate the game state accordingly. -To define a message, extend the Message class. +To define a message, declare a struct which implements the IMessage interface. -```ts -import { Message } from "encompass-ecs"; +```cs +using Encompass; -class MotionMessage extends Message { - public x: number; - public y: number; +public struct MotionMessage : IMessage { + public Vector2 motion; } ``` Messages are temporary and destroyed at the end of the frame. {{% notice notice %}} -Ok fine, since you asked, Messages actually live in an object pool so that they aren't garbage-collected at runtime. But you as the game developer don't have to worry about that. +Because structs are value types, we can create as many of them as we want without worrying about creating pressure on the garbage collector. Neato! {{% /notice %}} diff --git a/content/concepts/renderer.md b/content/concepts/renderer.md index 4732dee..3c99118 100644 --- a/content/concepts/renderer.md +++ b/content/concepts/renderer.md @@ -7,50 +7,91 @@ weight: 30 A Renderer is responsible for reading the game state and telling the game engine what to draw to the screen. {{% notice notice %}} -Remember: Encompass isn't a game engine and it doesn't have a rendering system. So Renderers aren't actually doing the rendering, they're just telling the game engine what to render. +Remember: Encompass isn't a game engine and it doesn't have a rendering system. So Renderers aren't actually doing the rendering,it is just a way to structure how we tell the game engine what to render. {{% /notice %}} -There are two kinds of renderers: GeneralRenderers and EntityRenderers. +There are two kinds of renderers: GeneralRenderers and OrderedRenderers. A GeneralRenderer is a Renderer which reads the game state in order to draw elements to the screen. It also requires a layer, which represents the order in which it will draw to the screen. -If you were using the LOVE engine, a GeneralRenderer might look like this: +If you were using MonoGame, a GeneralRenderer might look like this: -```ts -import { GeneralRenderer } from "encompass-ecs"; -import { ScoreComponent } from "game/components/score"; +```cs +using System; +using Encompass; +using Microsoft.Xna.Framework; +using SamuraiGunn2.Components; +using SamuraiGunn2.Editor.Components; +using SamuraiGunn2.Helpers; -export class ScoreRenderer extends GeneralRenderer { - public layer = 4; +namespace SamuraiGunn2.Editor.Renderers +{ + public class GridRenderer : GeneralRenderer + { + private int gridSize; + private PrimitiveDrawer primitiveDrawer; - public render() { - const score_component = this.read_component(ScoreComponent); + public GridRenderer(PrimitiveDrawer primitiveDrawer) + { + this.primitiveDrawer = primitiveDrawer; + this.gridSize = 16; + } - love.graphics.print(score_component.score, 20, 20); + public override void Render() + { + if (SomeComponent() && SomeComponent()) + { + var entity = ReadEntity(); + var transformComponent = GetComponent(entity); + + Rectangle rectangle = new Rectangle + { + X = (transformComponent.Position.X / gridSize) * gridSize, + Y = (transformComponent.Position.Y / gridSize) * gridSize, + Width = gridSize, + Height = gridSize + }; + + primitiveDrawer.DrawBorder(rectangle, 0, new System.Numerics.Vector2(1, 1), Color.White, 1); + } + } } } ``` -An EntityRenderer provides a structure for the common pattern of drawing an Entity which has a particular collection of Components and a specific type of DrawComponent. They also have the ability to draw DrawComponents at their specific layer. +GeneralRenderers are great for things like UI layers, where we always want a group of particular elements to be drawn at a specific layer regardless of the specifics of the game state. -If you were using the LOVE engine, a GeneralRenderer might look like this: +An OrderedRenderer provides a structure for the common pattern of wanting to draw an individual Component at a specific layer. OrderedRenderers must specify a component that implements IDrawableComponent. -```ts -import { EntityRenderer } from "encompass-ecs"; -import { PointComponent } from "game/components/point"; -import { PositionComponent } from "game/components/position"; +If you were using MonoGame, an OrderedRenderer might look like this: -@Renders(PointComponent, PositionComponent) -export class PointRenderer extends EntityRenderer { - public render(entity: Entity) { - const point_component = entity.get_component(PointComponent); - const position_component = entity.get_component(PositionComponent); +```cs +using Encompass; +using Microsoft.Xna.Framework.Graphics; +using SamuraiGunn2.Components; +using SamuraiGunn2.Extensions; +using System; +using System.Numerics; - const color = point_component.color; - love.graphics.setColor(color.r, color.g, color.b, color.a); - love.graphics.point(position_component.x, position_component.y); +namespace SamuraiGunn2.Renderers +{ + public class Texture2DRenderer : OrderedRenderer + { + private SpriteBatch spriteBatch; + + public Texture2DRenderer(SpriteBatch spriteBatch) + { + this.spriteBatch = spriteBatch; + } + + public override void Render(Entity entity, Texture2DComponent textureComponent) + { + var transformComponent = GetComponent(entity); + + spriteBatch.Draw(textureComponent.texture, transformComponent.Position, null, textureComponent.color, transformComponent.Rotation, textureComponent.origin, transformComponent.Scale, SpriteEffects.None, 0); + } } } ``` -For 2D games, you will need to use layers to be specific about the order in which entities draw. For a 3D game you will probably end up delegating rendering to some kind of scene/camera system. +For 2D games, you will need to use layers to be specific about the order in which elements are drawn to the screen. For a 3D game you will probably end up delegating most of the rendering to some kind of scene/camera system. diff --git a/content/concepts/world.md b/content/concepts/world.md index ce0ede4..518519b 100644 --- a/content/concepts/world.md +++ b/content/concepts/world.md @@ -6,32 +6,59 @@ weight: 100 World is the pie crust that contains all the delicious Encompass ingredients together. -The World's *update* function drives the simulation and should be controlled from your engine's update loop. +The World's *Update* function drives the simulation and should be controlled from your engine's update loop. -The World's *draw* function tells the Renderers to draw the scene. +The World's *Draw* function tells the Renderers to draw the scene. -In LÖVE, the starter project game loop looks like this: +In MonoGame, the game loop looks something like this: -```ts -export class Game { - private world: World; - private canvas: Canvas; +```cs +using Encompass; +using Microsoft.Xna.Framework; + +public class MyGame : Game +{ + private World world; + SpriteBatch spriteBatch; + + RenderTarget2D gameRenderTarget; + RenderTarget2D levelBrowserRenderTarget; + RenderTarget2D uiRenderTarget; ... - public update(dt: number) { - this.world.update(dt); + /// + /// Allows the game to run logic such as updating the world, + /// checking for collisions, gathering input, and playing audio. + /// + /// Provides a snapshot of timing values. + protected override void Update(GameTime gameTime) + { + if (GamePad.GetState(Microsoft.Xna.Framework.PlayerIndex.One).Buttons.Back == ButtonState.Pressed || Keyboard.GetState().IsKeyDown(Keys.Escape)) + Exit(); + + world.Update(gameTime.ElapsedGameTime.TotalSeconds); + + base.Update(gameTime); } - public draw() { - love.graphics.clear(); - love.graphics.setCanvas(this.canvas); - love.graphics.clear(); - this.world.draw(); - love.graphics.setCanvas(); - love.graphics.setBlendMode("alpha", "premultiplied"); - love.graphics.setColor(1, 1, 1, 1); - love.graphics.draw(this.canvas); + /// + /// This is called when the game should draw itself. + /// + /// Provides a snapshot of timing values. + protected override void Draw(GameTime gameTime) + { + world.Draw(); + + GraphicsDevice.SetRenderTarget(null); + + spriteBatch.Begin(SpriteSortMode.Deferred, null, SamplerState.PointClamp); + spriteBatch.Draw(gameRenderTarget, windowDimensions, Color.White); + spriteBatch.Draw(levelBrowserRenderTarget, windowDimensions, Color.White); + spriteBatch.Draw(uiRenderTarget, windowDimensions, Color.White); + spriteBatch.End(); + + base.Draw(gameTime); } } ``` @@ -46,4 +73,6 @@ Certain Encompass projects actually have multiple separate Worlds to manage cert *dt* stands for delta-time. Correct usage of delta-time is crucial to make sure that your game does not become *frame-dependent*, which is very bad. We'll talk more about frame-dependence later in the tutorial, but to briefly summarize, if your game is frame-dependent you will run into very frustrating behavior when running your game on different computer systems. +Even if you lock your game to a fixed timestep, writing your game with delta-time in mind can be the difference between changing the timestep being a one-line tweak or a weeks long hair-pulling nightmare. + That's it! Now that we have these high-level concepts down, let's build an actual, for-real game. diff --git a/content/concepts/world_builder.md b/content/concepts/world_builder.md index 851c30c..43b8ac0 100644 --- a/content/concepts/world_builder.md +++ b/content/concepts/world_builder.md @@ -6,45 +6,49 @@ weight: 35 WorldBuilder is used to construct a World from Engines, Renderers, and an initial state of Entities, Components, and Messages. -The WorldBuilder enforces certain rules about Engine structure. It is forbidden to have messages create cycles between Engines, and no Component may be mutated by more than one Engine. +The WorldBuilder enforces certain rules about Engine structure. It is forbidden to have messages create cycles between Engines, and no Component may be mutated by more than one Engine without declaring a priority. -The WorldBuilder uses Engines and their Message read/emit information to determine a valid ordering of the Engines, which is given to the World. +The WorldBuilder uses Engines and their Message read/send information to determine a valid ordering of the Engines, which is given to the World. -Here is an example usage: +Here is an example usage with MonoGame: -```ts -import { WorldBuilder } from "encompass-ecs"; -import { CanvasComponent } from "./components/canvas"; -import { PositionComponent } from "./components/position"; -import { VelocityComponent } from "./components/velocity"; -import { MotionEngine } from "./engines/motion"; -import { CanvasRenderer } from "./renderers/canvas"; +```cs +using Encompass; +using Microsoft.Xna.Framework; -class Game { +public class MyGame : Game +{ private World world; ... - public load() { - const world_builder = new WorldBuilder(); + protected override void LoadContent() + { + var worldBuilder = new WorldBuilder(); - world_builder.add_engine(MotionEngine); - world_builder.add_renderer(CanvasRenderer); + worldBuilder.AddEngine(new MotionEngine()); + worldBuilder.AddEngine(new TextureRenderer()); - const entity = world_builder.create_entity(); + var entity = worldBuilder.CreateEntity(); - const position_component = entity.add_component(PositionComponent); - position_component.x = 0; - position_component.y = 0; + SetComponent(entity, new PositionComponent + { + x = 0, + y = 0 + }); - const velocity_component = entity.add_component(VelocityComponent); - velocity_component.x = 20; - velocity_component.y = 0; + SetComponent(entity, new VelocityComponent + { + x = 20, + y = 0 + }); - const sprite_component = entity.add_component(SpriteComponent); - canvas_component.canvas = love.graphics.newImage("assets/sprites/ball.png"); + SetComponent(entity, new TextureComponent + { + texture = TextureHelper.LoadTexture("assets/sprites/ball.png") + }); - this.world = world_builder.build(); + world = worldBuilder.Build(); } ... @@ -54,5 +58,5 @@ class Game { Now our game will initialize with a ball that moves horizontally across the screen! {{% notice tip %}} -Make sure that you remember to add Engines to the WorldBuilder when you define them. Otherwise nothing will happen, which can be very embarrassing. +Be extra careful that you remember to add Engines to the WorldBuilder when you define them. Otherwise nothing will happen, which can be very embarrassing. {{% /notice %}} diff --git a/content/getting_started/choosing.md b/content/getting_started/choosing.md index 78d2a8e..fd11a34 100644 --- a/content/getting_started/choosing.md +++ b/content/getting_started/choosing.md @@ -16,17 +16,16 @@ That means it needs to run on top of an engine. Ultimately, this is a question that you have to answer for your project. Is there an engine you're already comfortable with? Which platforms are you targeting? Linux? PS4? Android? Are there any features you would really like to have, like a built-in physics simulator? These are questions that could help you choose an engine. -Encompass-TS can hook into any engine that supports JavaScript or Lua scripting. -Encompass-CS can hook into any engine that supports C# scripting. (**NOTE:** Encompass-CS is not done yet.) +Encompass-CS can hook into any engine that supports C# scripting. So you have a lot of choices! Here are some engines that I have used: -[LÖVE](https://love2d.org/) is a wonderful little framework for 2D games. It's cross-platform and very lightweight, but with a lot of powerful features, including a complete physics simulator. It uses Lua scripting, so you would want Encompass-TS. - [MonoGame](http://www.monogame.net/) is a cross-platform 2D/3D framework that thousands of games have used. You can use it to ship games on basically any platform that exists and it is extremely well-supported. It uses C# scripting. -[BabylonJS](https://www.babylonjs.com/) uses the power of WebGL to run 3D games in the browser. It has a powerful graphics pipeline and you can make stuff that has some wow factor. It runs on JS. +FNA + +Unity uses C# scripting, but you would have to adapt it in certain ways to the bastardized Unity architecture. I personally have never tried this but you are certainly welcome to give it a shot! Encompass gives you the power to develop using many different engines, so feel free to experiment and find one you like! And if you switch to an engine that uses the same scripting language, it's actually very easy to switch engines, because the simulation layer is mostly self-contained. diff --git a/content/getting_started/editor.md b/content/getting_started/editor.md index ab6c7eb..4b6c190 100644 --- a/content/getting_started/editor.md +++ b/content/getting_started/editor.md @@ -6,6 +6,6 @@ weight: 9 You will want some kind of text editor to develop Encompass projects. -I _highly_ recommend [VSCodium](https://vscodium.com/) if you are on Windows or OSX, and Code - OSS if you are on Linux. These are open-source distributions of Microsoft's VSCode editor, which features excellent Typescript integration and various convenient features, like an integrated Git interface and terminal. (Make sure you set the terminal to Git Bash if you are on Windows - this is under File -> Settings.) +I _highly_ recommend [VSCodium](https://vscodium.com/) if you are on Windows or OSX, and Code - OSS if you are on Linux. These are open-source distributions of Microsoft's VSCode editor, which features excellent C# integration and various convenient features, like an integrated Git interface and terminal. (Make sure you set the terminal to Git Bash if you are on Windows - this is under File -> Settings.) Of course, if you prefer some other editor, that will be perfectly fine. diff --git a/content/why/architecture/hyper_ecs.md b/content/why/architecture/hyper_ecs.md index 33ac870..3562212 100644 --- a/content/why/architecture/hyper_ecs.md +++ b/content/why/architecture/hyper_ecs.md @@ -10,28 +10,28 @@ The core of the architecture is the introduction of a new construct to ECS: the A Message is fundamentally a variant of Component, in that it only contains data. But, it is designed to be temporary and is discarded at the end of each frame. It is used to communicate useful information between Systems. -We also introduce some extra information to Systems. Each System must declare the Messages that it **Reads**, the Messages that it **Emits**, and the Components that it **Mutates**. +We also introduce some extra information to Systems. Each System must declare the Messages that it **Receives**, the Messages that it **Sends**, the Components that it **Reads** and the Components that it **Writes**. Let's go back to our earlier example. We have TransformComponent, which contains position and orientation data, and VelocityComponent, which contains an *x* and *y* component for linear motion. -Our MotionDetecterSystem reads each Entity that has both a TransformComponent and a VelocityComponent, and emits a MotionMessage, which contains a reference to the specific TransformComponent and the *x* and *y* velocity given by the VelocityComponent. +Our MotionDetecterSystem reads each Entity that has both a TransformComponent and a VelocityComponent, and emits a MotionMessage, which contains a reference to the Entity and the *x* and *y* velocity given by the VelocityComponent. -We also have a TeleportSystem that needs to teleport the character forward a bit. Let's say when the player presses the X button, a TeleportMessage is fired. The TeleportSystem reads this message and emits a MotionMessage in response. +We also have a **TeleportSystem** that needs to teleport the character forward a bit. Let's say when the player presses the X button, a TeleportMessage is fired. The TeleportSystem reads this message and emits a MotionMessage in response. -Now we have our MotionSystem. The MotionSystem declares that it Mutates the TransformComponent, reads the MotionMessages that apply to each TransformComponent, and applies them simultaneously, adding their *x* and *y* values to the TransformComponent. Voilà! No race conditions! And we can re-use similar behaviors easily without re-writing code by consolidating Messages. +Now we have our **MotionSystem**. The MotionSystem declares that it Mutates the TransformComponent, reads the MotionMessages that apply to each TransformComponent, and applies them simultaneously, adding their *x* and *y* values to the TransformComponent. Voilà! No race conditions! We can re-use similar behaviors easily without re-writing code by consolidating Messages. You might be wondering: how does the game know which order these systems need to be in? Well... **Hyper ECS figures it out for you.** -That's right! With the power of graph theory, we can construct an order for our Systems so that any System which Emits a certain Message runs before any System that Reads the same Message. This means, when you write behavior for your game, you *never* have to specify the order in which your Systems run. You simply write code, and the Systems run in a valid order, every time, without surprising you. +That's right! With the power of graph theory, we can construct an order for our Systems so that any System which Sends a certain Message runs before any System that Reads the same Message. This means, when you write behavior for your game, you *never* have to specify the order in which your Systems run. You simply write code, and the Systems run in a valid order, every time, without surprising you. Of course, to accomplish this, there are some restrictions that your Systems must follow. -Systems are not allowed to create message cycles. If System A emits Message B, which is read by System B which emits Message C, which is read by System A, then we cannot create a valid ordering of Systems. This is not a flaw in the architecture: A message cycle is simply evidence that you haven't quite thought through what your Systems are doing, and can generally be easily eliminated by the introduction of a new System. +Systems are not allowed to create message cycles. If System A sends Message B, which is read by System B which emits Message C, which is read by System A, then we cannot create a valid ordering of Systems. This is not a flaw in the architecture: A message cycle is simply evidence that you haven't quite thought through what your Systems are doing, and can generally be easily eliminated by the introduction of a new System. -Two separate systems are not allowed to Mutate the same Component. Obviously, if we allowed this, we would introduce the possibility of two Systems changing the same component, creating a race condition. If we have two Systems where it makes sense to change the same Component, we can create a new Message and System to consolidate the changes, and avoid race conditions. +The other restriction involves two separate systems which Write the same Component. They may do so, but they must declare a priority. Obviously, if we allowed Systems to Write components willy-nilly, we would introduce the possibility of two Systems changing the same component on the same frame, creating a race condition. If we have two Systems where it makes sense to change the same Component, we can either create a new Message and System to consolidate the changes, or we can declare write priority so that one System's changes always override the other's. This way we can avoid race conditions. -If you are used to programming games in an object-oriented way, you will likely find the ECS pattern counter-intuitive at first. But once you learn to think in a Hyper ECS way, you will be shocked at how flexible and simple your programs become. +If you are used to programming games in an object-oriented way, you will likely find the pattern counter-intuitive at first. But once you learn to think in a Hyper ECS way, you will be shocked at how flexible and simple your programs become. diff --git a/static/js/highlight.pack.js b/static/js/highlight.pack.js index 93adfb4..6d0a2f9 100644 --- a/static/js/highlight.pack.js +++ b/static/js/highlight.pack.js @@ -1,2 +1,2 @@ -/*! highlight.js v9.15.8 | BSD3 License | git.io/hljslicense */ -!function(e){var n="object"==typeof window&&window||"object"==typeof self&&self;"undefined"!=typeof exports?e(exports):n&&(n.hljs=e({}),"function"==typeof define&&define.amd&&define([],function(){return n.hljs}))}(function(a){var f=[],u=Object.keys,N={},c={},n=/^(no-?highlight|plain|text)$/i,s=/\blang(?:uage)?-([\w-]+)\b/i,t=/((^(<[^>]+>|\t|)+|(?:\n)))/gm,r={case_insensitive:"cI",lexemes:"l",contains:"c",keywords:"k",subLanguage:"sL",className:"cN",begin:"b",beginKeywords:"bK",end:"e",endsWithParent:"eW",illegal:"i",excludeBegin:"eB",excludeEnd:"eE",returnBegin:"rB",returnEnd:"rE",relevance:"r",variants:"v",IDENT_RE:"IR",UNDERSCORE_IDENT_RE:"UIR",NUMBER_RE:"NR",C_NUMBER_RE:"CNR",BINARY_NUMBER_RE:"BNR",RE_STARTERS_RE:"RSR",BACKSLASH_ESCAPE:"BE",APOS_STRING_MODE:"ASM",QUOTE_STRING_MODE:"QSM",PHRASAL_WORDS_MODE:"PWM",C_LINE_COMMENT_MODE:"CLCM",C_BLOCK_COMMENT_MODE:"CBCM",HASH_COMMENT_MODE:"HCM",NUMBER_MODE:"NM",C_NUMBER_MODE:"CNM",BINARY_NUMBER_MODE:"BNM",CSS_NUMBER_MODE:"CSSNM",REGEXP_MODE:"RM",TITLE_MODE:"TM",UNDERSCORE_TITLE_MODE:"UTM",COMMENT:"C",beginRe:"bR",endRe:"eR",illegalRe:"iR",lexemesRe:"lR",terminators:"t",terminator_end:"tE"},b="",h={classPrefix:"hljs-",tabReplace:null,useBR:!1,languages:void 0};function _(e){return e.replace(/&/g,"&").replace(//g,">")}function E(e){return e.nodeName.toLowerCase()}function v(e,n){var t=e&&e.exec(n);return t&&0===t.index}function l(e){return n.test(e)}function g(e){var n,t={},r=Array.prototype.slice.call(arguments,1);for(n in e)t[n]=e[n];return r.forEach(function(e){for(n in e)t[n]=e[n]}),t}function R(e){var a=[];return function e(n,t){for(var r=n.firstChild;r;r=r.nextSibling)3===r.nodeType?t+=r.nodeValue.length:1===r.nodeType&&(a.push({event:"start",offset:t,node:r}),t=e(r,t),E(r).match(/br|hr|img|input/)||a.push({event:"stop",offset:t,node:r}));return t}(e,0),a}function i(e){if(r&&!e.langApiRestored){for(var n in e.langApiRestored=!0,r)e[n]&&(e[r[n]]=e[n]);(e.c||[]).concat(e.v||[]).forEach(i)}}function m(o){function s(e){return e&&e.source||e}function c(e,n){return new RegExp(s(e),"m"+(o.cI?"i":"")+(n?"g":""))}!function n(t,e){if(!t.compiled){if(t.compiled=!0,t.k=t.k||t.bK,t.k){function r(t,e){o.cI&&(e=e.toLowerCase()),e.split(" ").forEach(function(e){var n=e.split("|");a[n[0]]=[t,n[1]?Number(n[1]):1]})}var a={};"string"==typeof t.k?r("keyword",t.k):u(t.k).forEach(function(e){r(e,t.k[e])}),t.k=a}t.lR=c(t.l||/\w+/,!0),e&&(t.bK&&(t.b="\\b("+t.bK.split(" ").join("|")+")\\b"),t.b||(t.b=/\B|\b/),t.bR=c(t.b),t.endSameAsBegin&&(t.e=t.b),t.e||t.eW||(t.e=/\B|\b/),t.e&&(t.eR=c(t.e)),t.tE=s(t.e)||"",t.eW&&e.tE&&(t.tE+=(t.e?"|":"")+e.tE)),t.i&&(t.iR=c(t.i)),null==t.r&&(t.r=1),t.c||(t.c=[]),t.c=Array.prototype.concat.apply([],t.c.map(function(e){return function(n){return n.v&&!n.cached_variants&&(n.cached_variants=n.v.map(function(e){return g(n,{v:null},e)})),n.cached_variants||n.eW&&[g(n)]||[n]}("self"===e?t:e)})),t.c.forEach(function(e){n(e,t)}),t.starts&&n(t.starts,e);var i=t.c.map(function(e){return e.bK?"\\.?(?:"+e.b+")\\.?":e.b}).concat([t.tE,t.i]).map(s).filter(Boolean);t.t=i.length?c(function(e,n){for(var t=/\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./,r=0,a="",i=0;i')+n+(t?"":b):n}function o(){E+=null!=l.sL?function(){var e="string"==typeof l.sL;if(e&&!N[l.sL])return _(g);var n=e?C(l.sL,g,!0,f[l.sL]):O(g,l.sL.length?l.sL:void 0);return 0")+'"');return g+=n,n.length||1}var s=B(e);if(!s)throw new Error('Unknown language: "'+e+'"');m(s);var a,l=t||s,f={},E="";for(a=l;a!==s;a=a.parent)a.cN&&(E=c(a.cN,"",!0)+E);var g="",R=0;try{for(var d,p,M=0;l.t.lastIndex=M,d=l.t.exec(n);)p=r(n.substring(M,d.index),d[0]),M=d.index+p;for(r(n.substr(M)),a=l;a.parent;a=a.parent)a.cN&&(E+=b);return{r:R,value:E,language:e,top:l}}catch(e){if(e.message&&-1!==e.message.indexOf("Illegal"))return{r:0,value:_(n)};throw e}}function O(t,e){e=e||h.languages||u(N);var r={r:0,value:_(t)},a=r;return e.filter(B).filter(M).forEach(function(e){var n=C(e,t,!1);n.language=e,n.r>a.r&&(a=n),n.r>r.r&&(a=r,r=n)}),a.language&&(r.second_best=a),r}function d(e){return h.tabReplace||h.useBR?e.replace(t,function(e,n){return h.useBR&&"\n"===e?"
":h.tabReplace?n.replace(/\t/g,h.tabReplace):""}):e}function o(e){var n,t,r,a,i,o=function(e){var n,t,r,a,i=e.className+" ";if(i+=e.parentNode?e.parentNode.className:"",t=s.exec(i))return B(t[1])?t[1]:"no-highlight";for(n=0,r=(i=i.split(/\s+/)).length;n/g,"\n"):n=e,i=n.textContent,r=o?C(o,i,!0):O(i),(t=R(n)).length&&((a=document.createElementNS("http://www.w3.org/1999/xhtml","div")).innerHTML=r.value,r.value=function(e,n,t){var r=0,a="",i=[];function o(){return e.length&&n.length?e[0].offset!==n[0].offset?e[0].offset"}function u(e){a+=""}function s(e){("start"===e.event?c:u)(e.node)}for(;e.length||n.length;){var l=o();if(a+=_(t.substring(r,l[0].offset)),r=l[0].offset,l===e){for(i.reverse().forEach(u);s(l.splice(0,1)[0]),(l=o())===e&&l.length&&l[0].offset===r;);i.reverse().forEach(c)}else"start"===l[0].event?i.push(l[0].node):i.pop(),s(l.splice(0,1)[0])}return a+_(t.substr(r))}(t,R(a),i)),r.value=d(r.value),e.innerHTML=r.value,e.className=function(e,n,t){var r=n?c[n]:t,a=[e.trim()];return e.match(/\bhljs\b/)||a.push("hljs"),-1===e.indexOf(r)&&a.push(r),a.join(" ").trim()}(e.className,o,r.language),e.result={language:r.language,re:r.r},r.second_best&&(e.second_best={language:r.second_best.language,re:r.second_best.r}))}function p(){if(!p.called){p.called=!0;var e=document.querySelectorAll("pre code");f.forEach.call(e,o)}}function B(e){return e=(e||"").toLowerCase(),N[e]||N[c[e]]}function M(e){var n=B(e);return n&&!n.disableAutodetect}return a.highlight=C,a.highlightAuto=O,a.fixMarkup=d,a.highlightBlock=o,a.configure=function(e){h=g(h,e)},a.initHighlighting=p,a.initHighlightingOnLoad=function(){addEventListener("DOMContentLoaded",p,!1),addEventListener("load",p,!1)},a.registerLanguage=function(n,e){var t=N[n]=e(a);i(t),t.aliases&&t.aliases.forEach(function(e){c[e]=n})},a.listLanguages=function(){return u(N)},a.getLanguage=B,a.autoDetection=M,a.inherit=g,a.IR=a.IDENT_RE="[a-zA-Z]\\w*",a.UIR=a.UNDERSCORE_IDENT_RE="[a-zA-Z_]\\w*",a.NR=a.NUMBER_RE="\\b\\d+(\\.\\d+)?",a.CNR=a.C_NUMBER_RE="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",a.BNR=a.BINARY_NUMBER_RE="\\b(0b[01]+)",a.RSR=a.RE_STARTERS_RE="!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~",a.BE=a.BACKSLASH_ESCAPE={b:"\\\\[\\s\\S]",r:0},a.ASM=a.APOS_STRING_MODE={cN:"string",b:"'",e:"'",i:"\\n",c:[a.BE]},a.QSM=a.QUOTE_STRING_MODE={cN:"string",b:'"',e:'"',i:"\\n",c:[a.BE]},a.PWM=a.PHRASAL_WORDS_MODE={b:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/},a.C=a.COMMENT=function(e,n,t){var r=a.inherit({cN:"comment",b:e,e:n,c:[]},t||{});return r.c.push(a.PWM),r.c.push({cN:"doctag",b:"(?:TODO|FIXME|NOTE|BUG|XXX):",r:0}),r},a.CLCM=a.C_LINE_COMMENT_MODE=a.C("//","$"),a.CBCM=a.C_BLOCK_COMMENT_MODE=a.C("/\\*","\\*/"),a.HCM=a.HASH_COMMENT_MODE=a.C("#","$"),a.NM=a.NUMBER_MODE={cN:"number",b:a.NR,r:0},a.CNM=a.C_NUMBER_MODE={cN:"number",b:a.CNR,r:0},a.BNM=a.BINARY_NUMBER_MODE={cN:"number",b:a.BNR,r:0},a.CSSNM=a.CSS_NUMBER_MODE={cN:"number",b:a.NR+"(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?",r:0},a.RM=a.REGEXP_MODE={cN:"regexp",b:/\//,e:/\/[gimuy]*/,i:/\n/,c:[a.BE,{b:/\[/,e:/\]/,r:0,c:[a.BE]}]},a.TM=a.TITLE_MODE={cN:"title",b:a.IR,r:0},a.UTM=a.UNDERSCORE_TITLE_MODE={cN:"title",b:a.UIR,r:0},a.METHOD_GUARD={b:"\\.\\s*"+a.UIR,r:0},a});hljs.registerLanguage("typescript",function(e){var r="[A-Za-z$_][0-9A-Za-z$_]*",t={keyword:"in if for while finally var new function do return void else break catch instanceof with throw case default try this switch continue typeof delete let yield const class public private protected get set super static implements enum export import declare type namespace abstract as from extends async await",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent encodeURI encodeURIComponent escape unescape Object Function Boolean Error EvalError InternalError RangeError ReferenceError StopIteration SyntaxError TypeError URIError Number Math Date String RegExp Array Float32Array Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require module console window document any number boolean string void Promise"},n={cN:"meta",b:"@"+r},a={b:"\\(",e:/\)/,k:t,c:["self",e.QSM,e.ASM,e.NM]},o={cN:"params",b:/\(/,e:/\)/,eB:!0,eE:!0,k:t,c:[e.CLCM,e.CBCM,n,a]};return{aliases:["ts"],k:t,c:[{cN:"meta",b:/^\s*['"]use strict['"]/},e.ASM,e.QSM,{cN:"string",b:"`",e:"`",c:[e.BE,{cN:"subst",b:"\\$\\{",e:"\\}"}]},e.CLCM,e.CBCM,{cN:"number",v:[{b:"\\b(0[bB][01]+)"},{b:"\\b(0[oO][0-7]+)"},{b:e.CNR}],r:0},{b:"("+e.RSR+"|\\b(case|return|throw)\\b)\\s*",k:"return throw case",c:[e.CLCM,e.CBCM,e.RM,{cN:"function",b:"(\\(.*?\\)|"+e.IR+")\\s*=>",rB:!0,e:"\\s*=>",c:[{cN:"params",v:[{b:e.IR},{b:/\(\s*\)/},{b:/\(/,e:/\)/,eB:!0,eE:!0,k:t,c:["self",e.CLCM,e.CBCM]}]}]}],r:0},{cN:"function",b:"function",e:/[\{;]/,eE:!0,k:t,c:["self",e.inherit(e.TM,{b:r}),o],i:/%/,r:0},{bK:"constructor",e:/\{/,eE:!0,c:["self",o]},{b:/module\./,k:{built_in:"module"},r:0},{bK:"module",e:/\{/,eE:!0},{bK:"interface",e:/\{/,eE:!0,k:"interface extends"},{b:/\$[(.]/},{b:"\\."+e.IR,r:0},n,a]}}); \ No newline at end of file +/*! highlight.js v9.16.2 | BSD3 License | git.io/hljslicense */ +!function(e){var n="object"==typeof window&&window||"object"==typeof self&&self;"undefined"==typeof exports||exports.nodeType?n&&(n.hljs=e({}),"function"==typeof define&&define.amd&&define([],function(){return n.hljs})):e(exports)}(function(a){var f=[],i=Object.keys,b={},u={},n=/^(no-?highlight|plain|text)$/i,l=/\blang(?:uage)?-([\w-]+)\b/i,t=/((^(<[^>]+>|\t|)+|(?:\n)))/gm,r={case_insensitive:"cI",lexemes:"l",contains:"c",keywords:"k",subLanguage:"sL",className:"cN",begin:"b",beginKeywords:"bK",end:"e",endsWithParent:"eW",illegal:"i",excludeBegin:"eB",excludeEnd:"eE",returnBegin:"rB",returnEnd:"rE",variants:"v",IDENT_RE:"IR",UNDERSCORE_IDENT_RE:"UIR",NUMBER_RE:"NR",C_NUMBER_RE:"CNR",BINARY_NUMBER_RE:"BNR",RE_STARTERS_RE:"RSR",BACKSLASH_ESCAPE:"BE",APOS_STRING_MODE:"ASM",QUOTE_STRING_MODE:"QSM",PHRASAL_WORDS_MODE:"PWM",C_LINE_COMMENT_MODE:"CLCM",C_BLOCK_COMMENT_MODE:"CBCM",HASH_COMMENT_MODE:"HCM",NUMBER_MODE:"NM",C_NUMBER_MODE:"CNM",BINARY_NUMBER_MODE:"BNM",CSS_NUMBER_MODE:"CSSNM",REGEXP_MODE:"RM",TITLE_MODE:"TM",UNDERSCORE_TITLE_MODE:"UTM",COMMENT:"C",beginRe:"bR",endRe:"eR",illegalRe:"iR",lexemesRe:"lR",terminators:"t",terminator_end:"tE"},_="",m={classPrefix:"hljs-",tabReplace:null,useBR:!1,languages:void 0},c="of and for in not or if then".split(" ");function C(e){return e.replace(/&/g,"&").replace(//g,">")}function E(e){return e.nodeName.toLowerCase()}function o(e){return n.test(e)}function s(e){var n,t={},r=Array.prototype.slice.call(arguments,1);for(n in e)t[n]=e[n];return r.forEach(function(e){for(n in e)t[n]=e[n]}),t}function g(e){var a=[];return function e(n,t){for(var r=n.firstChild;r;r=r.nextSibling)3===r.nodeType?t+=r.nodeValue.length:1===r.nodeType&&(a.push({event:"start",offset:t,node:r}),t=e(r,t),E(r).match(/br|hr|img|input/)||a.push({event:"stop",offset:t,node:r}));return t}(e,0),a}function d(e,n,t){var r=0,a="",i=[];function c(){return e.length&&n.length?e[0].offset!==n[0].offset?e[0].offset"}function l(e){a+=""}function o(e){("start"===e.event?u:l)(e.node)}for(;e.length||n.length;){var s=c();if(a+=C(t.substring(r,s[0].offset)),r=s[0].offset,s===e){for(i.reverse().forEach(l);o(s.splice(0,1)[0]),(s=c())===e&&s.length&&s[0].offset===r;);i.reverse().forEach(u)}else"start"===s[0].event?i.push(s[0].node):i.pop(),o(s.splice(0,1)[0])}return a+C(t.substr(r))}function R(n){return n.v&&!n.cached_variants&&(n.cached_variants=n.v.map(function(e){return s(n,{v:null},e)})),n.cached_variants?n.cached_variants:function e(n){return!!n&&(n.eW||e(n.starts))}(n)?[s(n,{starts:n.starts?s(n.starts):null})]:[n]}function v(e){if(r&&!e.langApiRestored){for(var n in e.langApiRestored=!0,r)e[n]&&(e[r[n]]=e[n]);(e.c||[]).concat(e.v||[]).forEach(v)}}function p(n,r){var a={};return"string"==typeof n?t("keyword",n):i(n).forEach(function(e){t(e,n[e])}),a;function t(t,e){r&&(e=e.toLowerCase()),e.split(" ").forEach(function(e){var n=e.split("|");a[n[0]]=[t,function(e,n){return n?Number(n):function(e){return-1!=c.indexOf(e.toLowerCase())}(e)?0:1}(n[0],n[1])]})}}function O(r){function s(e){return e&&e.source||e}function f(e,n){return new RegExp(s(e),"m"+(r.cI?"i":"")+(n?"g":""))}function a(a){var i,e,c={},u=[],l={},t=1;function n(e,n){c[t]=e,u.push([e,n]),t+=function(e){return new RegExp(e.toString()+"|").exec("").length-1}(n)+1}for(var r=0;r')+n+(t?"":_)}function l(){R+=null!=g.sL?function(){var e="string"==typeof g.sL;if(e&&!b[g.sL])return C(v);var n=e?x(g.sL,v,!0,d[g.sL]):B(v,g.sL.length?g.sL:void 0);return 0")+'"');if("end"===n.type){var r=function(e){var n=e[0],t=c(g,n);if(t){var r=g;for(r.skip?v+=n:(r.rE||r.eE||(v+=n),l(),r.eE&&(v=n));g.cN&&(R+=_),g.skip||g.sL||(p+=g.relevance),(g=g.parent)!==t.parent;);return t.starts&&(t.endSameAsBegin&&(t.starts.eR=t.eR),o(t.starts)),r.rE?0:n.length}}(n);if(null!=r)return r}return v+=t,t.length}var E=S(e);if(!E)throw new Error('Unknown language: "'+e+'"');O(E);var r,g=n||E,d={},R="";for(r=g;r!==E;r=r.parent)r.cN&&(R=u(r.cN,"",!0)+R);var v="",p=0;try{for(var M,N,h=0;g.t.lastIndex=h,M=g.t.exec(a);)N=t(a.substring(h,M.index),M),h=M.index+N;for(t(a.substr(h)),r=g;r.parent;r=r.parent)r.cN&&(R+=_);return{relevance:p,value:R,i:!1,language:e,top:g}}catch(e){if(e.message&&-1!==e.message.indexOf("Illegal"))return{i:!0,relevance:0,value:C(a)};throw e}}function B(t,e){e=e||m.languages||i(b);var r={relevance:0,value:C(t)},a=r;return e.filter(S).filter(T).forEach(function(e){var n=x(e,t,!1);n.language=e,n.relevance>a.relevance&&(a=n),n.relevance>r.relevance&&(a=r,r=n)}),a.language&&(r.second_best=a),r}function M(e){return m.tabReplace||m.useBR?e.replace(t,function(e,n){return m.useBR&&"\n"===e?"
":m.tabReplace?n.replace(/\t/g,m.tabReplace):""}):e}function N(e){var n,t,r,a,i,c=function(e){var n,t,r,a,i=e.className+" ";if(i+=e.parentNode?e.parentNode.className:"",t=l.exec(i))return S(t[1])?t[1]:"no-highlight";for(n=0,r=(i=i.split(/\s+/)).length;n/g,"\n"):n=e,i=n.textContent,r=c?x(c,i,!0):B(i),(t=g(n)).length&&((a=document.createElementNS("http://www.w3.org/1999/xhtml","div")).innerHTML=r.value,r.value=d(t,g(a),i)),r.value=M(r.value),e.innerHTML=r.value,e.className=function(e,n,t){var r=n?u[n]:t,a=[e.trim()];return e.match(/\bhljs\b/)||a.push("hljs"),-1===e.indexOf(r)&&a.push(r),a.join(" ").trim()}(e.className,c,r.language),e.result={language:r.language,re:r.relevance},r.second_best&&(e.second_best={language:r.second_best.language,re:r.second_best.relevance}))}function h(){if(!h.called){h.called=!0;var e=document.querySelectorAll("pre code");f.forEach.call(e,N)}}function S(e){return e=(e||"").toLowerCase(),b[e]||b[u[e]]}function T(e){var n=S(e);return n&&!n.disableAutodetect}return a.highlight=x,a.highlightAuto=B,a.fixMarkup=M,a.highlightBlock=N,a.configure=function(e){m=s(m,e)},a.initHighlighting=h,a.initHighlightingOnLoad=function(){addEventListener("DOMContentLoaded",h,!1),addEventListener("load",h,!1)},a.registerLanguage=function(n,e){var t=b[n]=e(a);v(t),t.rawDefinition=e.bind(null,a),t.aliases&&t.aliases.forEach(function(e){u[e]=n})},a.listLanguages=function(){return i(b)},a.getLanguage=S,a.autoDetection=T,a.inherit=s,a.IR=a.IDENT_RE="[a-zA-Z]\\w*",a.UIR=a.UNDERSCORE_IDENT_RE="[a-zA-Z_]\\w*",a.NR=a.NUMBER_RE="\\b\\d+(\\.\\d+)?",a.CNR=a.C_NUMBER_RE="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",a.BNR=a.BINARY_NUMBER_RE="\\b(0b[01]+)",a.RSR=a.RE_STARTERS_RE="!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~",a.BE=a.BACKSLASH_ESCAPE={b:"\\\\[\\s\\S]",relevance:0},a.ASM=a.APOS_STRING_MODE={cN:"string",b:"'",e:"'",i:"\\n",c:[a.BE]},a.QSM=a.QUOTE_STRING_MODE={cN:"string",b:'"',e:'"',i:"\\n",c:[a.BE]},a.PWM=a.PHRASAL_WORDS_MODE={b:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/},a.C=a.COMMENT=function(e,n,t){var r=a.inherit({cN:"comment",b:e,e:n,c:[]},t||{});return r.c.push(a.PWM),r.c.push({cN:"doctag",b:"(?:TODO|FIXME|NOTE|BUG|XXX):",relevance:0}),r},a.CLCM=a.C_LINE_COMMENT_MODE=a.C("//","$"),a.CBCM=a.C_BLOCK_COMMENT_MODE=a.C("/\\*","\\*/"),a.HCM=a.HASH_COMMENT_MODE=a.C("#","$"),a.NM=a.NUMBER_MODE={cN:"number",b:a.NR,relevance:0},a.CNM=a.C_NUMBER_MODE={cN:"number",b:a.CNR,relevance:0},a.BNM=a.BINARY_NUMBER_MODE={cN:"number",b:a.BNR,relevance:0},a.CSSNM=a.CSS_NUMBER_MODE={cN:"number",b:a.NR+"(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?",relevance:0},a.RM=a.REGEXP_MODE={cN:"regexp",b:/\//,e:/\/[gimuy]*/,i:/\n/,c:[a.BE,{b:/\[/,e:/\]/,relevance:0,c:[a.BE]}]},a.TM=a.TITLE_MODE={cN:"title",b:a.IR,relevance:0},a.UTM=a.UNDERSCORE_TITLE_MODE={cN:"title",b:a.UIR,relevance:0},a.METHOD_GUARD={b:"\\.\\s*"+a.UIR,relevance:0},a});hljs.registerLanguage("cs",function(e){var a={keyword:"abstract as base bool break byte case catch char checked const continue decimal default delegate do double enum event explicit extern finally fixed float for foreach goto if implicit in int interface internal is lock long object operator out override params private protected public readonly ref sbyte sealed short sizeof stackalloc static string struct switch this try typeof uint ulong unchecked unsafe ushort using virtual void volatile while add alias ascending async await by descending dynamic equals from get global group into join let nameof on orderby partial remove select set value var when where yield",literal:"null false true"},i={cN:"number",v:[{b:"\\b(0b[01']+)"},{b:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{b:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)"}],relevance:0},c={cN:"string",b:'@"',e:'"',c:[{b:'""'}]},r=e.inherit(c,{i:/\n/}),n={cN:"subst",b:"{",e:"}",k:a},t=e.inherit(n,{i:/\n/}),s={cN:"string",b:/\$"/,e:'"',i:/\n/,c:[{b:"{{"},{b:"}}"},e.BE,t]},l={cN:"string",b:/\$@"/,e:'"',c:[{b:"{{"},{b:"}}"},{b:'""'},n]},b=e.inherit(l,{i:/\n/,c:[{b:"{{"},{b:"}}"},{b:'""'},t]});n.c=[l,s,c,e.ASM,e.QSM,i,e.CBCM],t.c=[b,s,r,e.ASM,e.QSM,i,e.inherit(e.CBCM,{i:/\n/})];var o={v:[l,s,c,e.ASM,e.QSM]},d=e.IR+"(<"+e.IR+"(\\s*,\\s*"+e.IR+")*>)?(\\[\\])?";return{aliases:["csharp","c#"],k:a,i:/::/,c:[e.C("///","$",{rB:!0,c:[{cN:"doctag",v:[{b:"///",relevance:0},{b:"\x3c!--|--\x3e"},{b:""}]}]}),e.CLCM,e.CBCM,{cN:"meta",b:"#",e:"$",k:{"meta-keyword":"if else elif endif define undef warning error line region endregion pragma checksum"}},o,i,{bK:"class interface",e:/[{;=]/,i:/[^\s:,]/,c:[e.TM,e.CLCM,e.CBCM]},{bK:"namespace",e:/[{;=]/,i:/[^\s:]/,c:[e.inherit(e.TM,{b:"[a-zA-Z](\\.?\\w)*"}),e.CLCM,e.CBCM]},{cN:"meta",b:"^\\s*\\[",eB:!0,e:"\\]",eE:!0,c:[{cN:"meta-string",b:/"/,e:/"/}]},{bK:"new return throw await else",relevance:0},{cN:"function",b:"("+d+"\\s+)+"+e.IR+"\\s*\\(",rB:!0,e:/\s*[{;=]/,eE:!0,k:a,c:[{b:e.IR+"\\s*\\(",rB:!0,c:[e.TM],relevance:0},{cN:"params",b:/\(/,e:/\)/,eB:!0,eE:!0,k:a,relevance:0,c:[o,i,e.CBCM]},e.CLCM,e.CBCM]}]}}); \ No newline at end of file diff --git a/themes/hugo-theme-learn/static/css/hugo-theme.css b/themes/hugo-theme-learn/static/css/hugo-theme.css index 741cab1..95ace51 100644 --- a/themes/hugo-theme-learn/static/css/hugo-theme.css +++ b/themes/hugo-theme-learn/static/css/hugo-theme.css @@ -250,5 +250,5 @@ figcaption h4 { } .is-sticky #top-bar { - box-shadow: -1px 2px 5px 1px rgba(0, 0, 0, 0.1); -} \ No newline at end of file + box-shadow: -1px 2px 5px 1px rgba(0, 0, 0, 0.1); +} diff --git a/themes/hugo-theme-learn/static/css/theme.css b/themes/hugo-theme-learn/static/css/theme.css index 2e1f65e..3379b1d 100644 --- a/themes/hugo-theme-learn/static/css/theme.css +++ b/themes/hugo-theme-learn/static/css/theme.css @@ -26,6 +26,20 @@ font-style: normal; font-weight: 200; } +@font-face { + font-family: 'Public Sans'; + src: url("../fonts/publicsans-regular-webfont.woff") format("woff"); + src: url("../fonts/publicsans-regular-webfont.woff2") format("woff2"); + font-style: normal; + font-weight: 400; +} +@font-face { + font-family: 'Public Sans'; + src: url("../fonts/publicsans-light-webfont.woff") format("woff"); + src: url("../fonts/publicsans-light-webfont.woff2") format("woff2"); + font-style: normal; + font-weight: 300; +} @font-face { font-family: 'Work Sans'; font-style: normal; @@ -448,10 +462,10 @@ textarea:focus, input[type="email"]:focus, input[type="number"]:focus, input[typ margin: 0; } body { - font-family: "Work Sans", "Helvetica", "Tahoma", "Geneva", "Arial", sans-serif; + font-family: "Public Sans", "Helvetica", "Tahoma", "Geneva", "Arial", sans-serif; font-weight: 300; line-height: 1.6; - font-size: 18px !important; + font-size: 16px !important; } h2, h3, h4, h5, h6 { font-family: "Work Sans", "Helvetica", "Tahoma", "Geneva", "Arial", sans-serif; diff --git a/themes/hugo-theme-learn/static/fonts/publicsans-light-webfont.woff b/themes/hugo-theme-learn/static/fonts/publicsans-light-webfont.woff new file mode 100644 index 0000000..4dff0ea Binary files /dev/null and b/themes/hugo-theme-learn/static/fonts/publicsans-light-webfont.woff differ diff --git a/themes/hugo-theme-learn/static/fonts/publicsans-light-webfont.woff2 b/themes/hugo-theme-learn/static/fonts/publicsans-light-webfont.woff2 new file mode 100644 index 0000000..3102ef1 Binary files /dev/null and b/themes/hugo-theme-learn/static/fonts/publicsans-light-webfont.woff2 differ diff --git a/themes/hugo-theme-learn/static/fonts/publicsans-regular-webfont.woff b/themes/hugo-theme-learn/static/fonts/publicsans-regular-webfont.woff new file mode 100644 index 0000000..f08bcf5 Binary files /dev/null and b/themes/hugo-theme-learn/static/fonts/publicsans-regular-webfont.woff differ diff --git a/themes/hugo-theme-learn/static/fonts/publicsans-regular-webfont.woff2 b/themes/hugo-theme-learn/static/fonts/publicsans-regular-webfont.woff2 new file mode 100644 index 0000000..19bddb6 Binary files /dev/null and b/themes/hugo-theme-learn/static/fonts/publicsans-regular-webfont.woff2 differ