Click here to Skip to main content
16,003,902 members
Articles / Desktop Programming / WPF

WPF based exciting Pong Game

Rate me:
Please Sign up or sign in to vote.
5.00/5 (10 votes)
19 Jul 2024MIT9 min read 41K   32   15
This is a multi player Pong game written in C#/ WPF using Visual Studio
Pongs is a 2 player game in which 2 players move their paddles simultaneously to hit a ball back and forth. Player 1 uses keys 'W', 'A', 'S', and 'D' for up, down, left and right respectively and Player 2 uses arrow keys. Both players attempt to hit the ball to opponents side. If the ball goes past a player's side of the screen, their opponent gains a point.

New Update: Screen Shake Added

Quick Demo: click here

Source Code: click here

Description

Pongs is a game developed in a weeks time where you can move your paddle using the keys W for up movement, A, S, D (likewise) and arrow keys to hit the ball to the other side.

Image 1
Figure 1 - Gif of game being played

If the ball goes past your side of the screen, the other person gets a point. The game goes on forever and keeps track of your score.

Image 2
Figure 2 - Directions page from game

Settings

There is a settings page in this game. The settings are...

Image 3
Figure 3 - Settings page from game
  • Ball Speed - Changes the speed at which the ball moves
  • Ball Size - Changes the size of the ball
  • Paddle Speed - Changes the speed at which the paddles move
  • Paddle Size - Changes the size of the paddles
  • Rounds to Win - The amount of wins a player needs to win the game

Ball/Paddle/Wall/Background Colors - The different shapes in the game that can have their color changed. Changing the colors on the settings page changes the colors of the board in real time:

Image 4

The color picker on the settings page also changes color when you change the color of the game.

Image 5

These specific colors make the game look like this:

Image 6

There are also pause and restart buttons which either pause the game until clicked again or restart the game respectively

Image 7
Figure 4 - Pause and restart button from game

Code Explanation

Drawing the board

One key part of my code is drawing everything on the board, such as the paddle and the ball. This is done through 3 different functions.

        public void DrawPaddle(Rectangle Paddle, double x, double y)
        {
            PaddleColor = new SolidColorBrush(SliderInfo.PaddleColor);

            Paddle.Width = 1.5 * SliderInfo.PaddleSize;
            Paddle.Height = 7.8 * SliderInfo.PaddleSize;
            [code deleted for brevity]

            Canvas.SetTop(Paddle, y);
            Canvas.SetLeft(Paddle, x);
            UpdateLocations("paddle1");
            UpdateLocations("paddle2");
        }
The above code draws the paddle onto the board. It is drawn separate from the other shapes since it is the only shape that moves via player input, and also requires inputs, such as which paddle and the x and y coordinates of it. The UpdateLocations function updates the variables P1Left, P1Right, P1Up, P1Down, and their counterparts depending on the string given.
        public void ReDraw()
        {     
            [code deleted for brevity]

            if (WindowState == WindowState.Maximized)
            {
                Window.Height = (int)System.Windows.SystemParameters.PrimaryScreenHeight;
                Window.Width = (int)System.Windows.SystemParameters.PrimaryScreenWidth;
            }

            Board.Width = Window.Width;
            Board.Height = Window.Height;

            sbkGameEngine.y1 = Board.Height / 2 - (paddle1.Height / 2);
            sbkGameEngine.y2 = sbkGameEngine.y1;
            sbkGameEngine.x2 = Board.Width - 32;
            sbkGameEngine.x1 = 0;

            Ball.Width = 1.5 * SliderInfo.BallSize;
            Ball.Height = 1.5 * SliderInfo.BallSize;
            [code deleted for brevity]
            Canvas.SetTop(Ball, Board.Height / 2 - Ball.Height / 2);
            Canvas.SetLeft(Ball, Board.Width / 2 - Ball.Width / 2);
            UpdateLocations("ball");

            Menu.Width = Board.Width;
            if (Board.Width - (SettingsMenu.Width + About.Width + Help.Width) - (1.5 * RestartButton.Width + PauseButton.Width) - 2 > 0)
            {
                Spacer.Width = Board.Width - (SettingsMenu.Width + About.Width + Help.Width) - (1.5 * RestartButton.Width + PauseButton.Width) - 2;
            }

            P1Scoreboard.Text = "" + sbkGameEngine.P1Score;
            P2Scoreboard.Text = "" + sbkGameEngine.P2Score;

            sbkGameEngine.CanBallMove = false;
            ReDrawUnmoving();
        }

This function draws all of the unmoving shapes, as well as the menu and board. This is done to initially set up the board and to reset the board when necessary.

        private void ReDrawUnmoving()
        {
            Boundary.Stroke = WallColor;
            [code deleted for brevity]
            Boundary.Y1 = 0;
            Boundary.Y2 = Board.Height;

            BottomWall.Width = Board.Width;
            BottomWall.Height = 24;
            [code deleted for brevity]
            Canvas.SetTop(BottomWall, Board.Height - 63);
            Canvas.SetLeft(BottomWall, 0);

            Menu.BorderBrush = WallColor;
            Spacer.BorderBrush = WallColor;
            Menu.Background = WallColor;
            Spacer.Background = WallColor;

            Board.Background = BackgroundColor;
        }

This function does the same thing as ReDraw(), just this time drawing the unmoving parts.

These functions allow for the board to be drawn, which a is key element of my game. This is because knowing where everything is on the board is necessary for the player to react in the right way.

Moving the Ball

Another key part of my game is the movement of the ball. This is because the main point of the game is to try and block the ball from moving into your side of the field.

public void BallMovement()
        {
            if (sbkGameEngine.CanBallMove && sbkGameEngine.GamePlayable)
            {
                Canvas.SetTop(Ball, Canvas.GetTop(Ball) + sbkGameEngine.VMovement);
                Canvas.SetLeft(Ball, Canvas.GetLeft(Ball) + sbkGameEngine.HMovement);
                UpdateLocations("ball");
            }
            if (sbkGameEngine.P1Wins)
            {
                WhoWon_.Text = "Player 1 Wins!";
                WhoWon_.Visibility = Visibility.Visible;
                RestartText.Visibility = Visibility.Visible;
                OnPause(Ball, a);
                sbkGameEngine.i = 2;
            }
            if (sbkGameEngine.P2Wins)
            {
                WhoWon_.Text = "Player 2 Wins!";
                WhoWon_.Visibility = Visibility.Visible;
                RestartText.Visibility = Visibility.Visible;
                OnPause(Ball, a);
                sbkGameEngine.i = 2;
            }
        }

This function only takes the variables from the sbkGameEngine class to move the ball. It doesn't do any calculations itself and just does what the engine tells it to do via the variables that are changed by the sbkGameEngine.

public void BallMovement()
        {
            log.Info("BallMovement Start");
            if (GamePlayable)
            {
                int P1Top = Game.P1Up;
                int P1Bottom = Game.P1Down;
                int P1Left = Game.P1Left;
                int P1Right = Game.P1Right;
                int P2Top = Game.P2Up;
                int P2Bottom = Game.P2Down;
                int P2Left = Game.P2Left;
                int P2Right = Game.P2Right;
                int BallTop = Game.BallUp;
                int BallBottom = Game.BallDown;
                int BallLeft = Game.BallLeft;
                int BallRight = Game.BallRight;

                if ((P2Bottom > BallTop && P2Top < BallBottom && BallLeft < P2Right && BallRight > P2Left && HMovement == 1) || (P1Bottom > BallTop && P1Top < BallBottom && BallLeft < P1Right && BallRight > P1Left && HMovement == -1))
                {
                    HMovement *= -1;
                    Console.Beep(37, 10);
                }
                if (BoundaryCheck(Game.Ball, 25, (int)(Game.Height - (Game.BottomWall.Height * 2) - 5), 0, 0, true, true, false, false) == false)
                {
                    VMovement *= -1;
                    Console.Beep(70, 5);
                }
                if (BallLeft >= mGuiReference.Board.Width)
                {
                    P1Score++;
                    Game.ReDraw();
                    log.Info("Player 1 scored!");
                    if (P1Score == SliderInfo.RoundsToWin)
                    {
                        P1Wins = true;
                    }
                }
                if (BallLeft + 15 <= 0)
                {
                    P2Score++;
                    Game.ReDraw();
                    log.Info("Player 2 scored!");
                    if (P2Score == SliderInfo.RoundsToWin)
                    {
                        P2Wins = true;
                    }
                }
            }
            log.Info("BallMovement End");
        }

This function is the one actually doing the calculations. It changes the variables controlling which direction the ball is moving when certain conditions are met, like swapping directions after hitting a paddle.

Both these functions combine to move the ball around and allow for the ball to interact with it's environment.

Paddle Movement

Another element necessary to the creation of the game is the movement of the paddles, since it is the only player controlled shape.

public void OnKeyDown(object sender, KeyEventArgs e)
        {      
            if (AllowedKeys.Contains(e.Key))
            {
                if (i == 0)
                {
                    KeysPressed.Add(e.Key);
                    CanBallMove = true;
                }
            }
        }
This function, when a key is pressed, adds that key to a hashset. The hashset will be later used to identify which keys are being pressed and which are not.
public void OnKeyUp(object sender, KeyEventArgs e)
        {
            if (KeysPressed.Contains(e.Key))
            {
                KeysPressed.Remove(e.Key);
            }
        }
This function removes the let go key and runs when a key is pressed.
public void PressedKeys(/*object? sender, EventArgs e*/)
        {
            if (GamePlayable)
            {
                if (KeysPressed.Contains(Key.Up) && BoundaryCheck(Game.paddle2, 25, (int)mGuiReference.Board.Height - 78, (int)mGuiReference.Board.Width / 2, (int)mGuiReference.Board.Width, true, false, false, false) == true)
                {
                    y2 -= 2;
                }
                if (KeysPressed.Contains(Key.W) && BoundaryCheck(Game.paddle1, 25, (int)mGuiReference.Board.Height - 78, (int)mGuiReference.Board.Width / 2, (int)mGuiReference.Board.Width, true, false, false, false) == true)
                {
                    y1 -= 2;
                }

                if (KeysPressed.Contains(Key.Down) && BoundaryCheck(Game.paddle2, 25, (int)mGuiReference.Board.Height - 78, (int)mGuiReference.Board.Width / 2, (int)mGuiReference.Board.Width, false, true, false, false) == true)
                {
                    y2 += 2;
                }
                if (KeysPressed.Contains(Key.S) && BoundaryCheck(Game.paddle1, 25, (int)mGuiReference.Board.Height - 78, (int)mGuiReference.Board.Width / 2, (int)mGuiReference.Board.Width, false, true, false, false) == true)
                {
                    y1 += 2;
                }

                if (KeysPressed.Contains(Key.Left) && BoundaryCheck(Game.paddle2, 25, (int)mGuiReference.Board.Height - 78, (int)mGuiReference.Board.Width / 2, (int)mGuiReference.Board.Width / 2, false, false, true, false) == true)
                {
                    x2 -= 2;
                }
                if (KeysPressed.Contains(Key.A) && BoundaryCheck(Game.paddle1, 25, (int)mGuiReference.Board.Height - 78, 0, (int)mGuiReference.Board.Width / 2, false, false, true, false) == true)
                {
                    x1 -= 2;
                }

                if (KeysPressed.Contains(Key.Right) && BoundaryCheck(Game.paddle2, 25, (int)mGuiReference.Board.Height - 78, 0, (int)mGuiReference.Board.Width - 20, false, false, false, true) == true)
                {
                    x2 += 2;
                }
                if (KeysPressed.Contains(Key.D) && BoundaryCheck(Game.paddle1, 25, (int)mGuiReference.Board.Height - 78, 0, (int)mGuiReference.Board.Width / 2, false, false, false, true) == true)
                {
                    x1 += 2;
                }
            }
        }        

The above function is run repeatedly by a different function and checks if any keys are in the hashset. If a key is in the hashset, it will do the corresponding move, such as moving the player 2 paddle up when the up key is in the hashset.

These all combine to allow player key inputs to be able to correspond to actions taken by the paddles in the game. 

Settings Page

The settings page requires sliders that change variables upon the slider changing so that whatever the slider says to happen happens. This is done using XAML and data binding.

    public static double BallSpeed { get; set; }
    public static double BallSize { get; set; }

    public static Color PaddleColor { get; set; }
    public static Color WallColor { get; set; }

The slider info class contains some of the variables for the sliders to act upon. These variables will be the ones that change when sliders are shifted.

<Slider x:Name="BallSpeedSlider" Margin="136,72,0,0" Maximum="9" Minimum="2" IsSnapToTickEnabled="True" Value="{Binding BallSpeed, Mode=TwoWay}" ValueChanged="OnValueChanged" HorizontalAlignment="Left" Width="226" Height="123" VerticalAlignment="Top" Grid.ColumnSpan="2"/>

<xctk:ColorPicker x:Name="Wall_Color_Picker"  Margin="207,339,0,0" Height="23" VerticalAlignment="Top" HorizontalAlignment="Left" Width="101" ShowDropDownButton = "False" ShowTabHeaders="False" ColorMode="ColorCanvas" SelectedColor="{Binding WallColor, Mode=TwoWay}" Grid.Column="1"/>

The above are examples of sliders and color pickers made in XAML that change the variables using data binding. Data binding binds the value of a slider to a variable, and setting the bode of the binding to TwoWay allows for when the slider is changed, the variable is too and vice versa.

These variables are connected to attributes like the speed of the ball or the color of the paddle such that when a slider or color picker is changed, the attribute changes as well. This allows the settings page to function in an uncomplex manner.

Data Binding

SelectedColor="{Binding WallColor, Mode=TwoWay}"

The above is a sample of color binding in XAML. SelectedColor is the color that was selected on the color picker. You can see that it's bound to WallColor, as in the curly brackets it says Binding WallColor. WallColor can be replaced with any other variable and the color picker will instead be bound to that variable, assuming that that variable accepts an color input. The Mode being TwoWay allows for changes in the color picker to change the variable and changes in the variable to change the color picker. There are 4 types of data binding.

1. OneWay - Only changes in the source property (such as a color picker or slider) affect the variable

2. TwoWay - Changes in the variable affect the source property and changes in the source property affect the variable

3. OneWayToSource - Only changes in the variable affect the source property

4. OneTime - Only updates the source property once when initializing the application using the variable

These 4 types all have their own use cases, but I only every used TwoWay in my projects, as the others were not needed for any of my game's functionalities.

Shaking the Screen

By shaking the screen when a player scores a point, it creates visual feedback for the players, allowing them to feel a higher sense of accomplishment. While seeming easy to code at first, it comes with several obstacles. 

To create this effect, I used storyboarding since it was a simple way to make the animation play out. The storyboard looks like this.

<Storyboard RepeatBehavior="0:0:1" Name="DownRight">
     <DoubleAnimation Storyboard.TargetName="Window" 
     Storyboard.TargetProperty="(Window.Left)" From="{Binding WindowLeft, Mode=TwoWay}" To="{Binding Path=WindowLeftTo, Mode=TwoWay}" 
     Duration="0:0:0:0.03" BeginTime="0:0:0" AutoReverse="true" RepeatBehavior="3x" FillBehavior="Stop"/>
     <DoubleAnimation Storyboard.TargetName="Window" 
     Storyboard.TargetProperty="(Window.Top)" From="{Binding WindowTop, Mode=TwoWay}" To="{Binding Path=WindowTopTo, Mode=TwoWay}" 
     Duration="0:0:0:0.03" BeginTime="0:0:0" AutoReverse="true" RepeatBehavior="3x" FillBehavior="Stop"/>
</Storyboard>

This storyboard moves the window up and down as well as left to right using two double animations. The double animations require the inputs From, the place where the window should start the animation, and to, where the window should end the animation. I used the window's left and top values for this, and for the to I used a variable that was the left of the window + 10 and the top of the window + 10. 

To get these values, you may think that you could use "Application.Current.MainWindow.Left" and "Application.Current.MainWindow.Top". However, due to the way WPF was made, if the window was never moves, it would return NaN. This is important because this makes the code crash when the animation was run if the window was never moved. Because of this, I had to use a different method to obtain the coordinates of the window.

        [StructLayout(LayoutKind.Sequential)]
        public struct RECT
        {
            public int Left;
            public int Top;
            public int Right;
            public int Bottom;
        }

        [DllImport("user32.dll")]
        [return: MarshalAs(UnmanagedType.Bool)]
        static extern bool GetWindowRect(IntPtr hWnd, out RECT lpRect);

        public event PropertyChangedEventHandler PropertyChanged; 

This code creates a rect that gains the coordinates of the window. Thus allows for the code to find the coordinates of the rect, which is the same as the coordinates of the window. and use for the from and to parameters in the storyboard.

To get the new coordinates of the window, when the window opens or is moved, it calls the following function.

        private void GetNewLocation()
        {
            RECT rect;

            IntPtr windowHandle = new WindowInteropHelper(Window).Handle;

            GetWindowRect(Process.GetCurrentProcess().MainWindowHandle, out rect);

            WindowLeft = rect.Left;
            WindowTop = rect.Top;
            PropertyChanged(this, new PropertyChangedEventArgs(nameof(WindowLeft)));
            PropertyChanged(this, new PropertyChangedEventArgs(nameof(WindowLeftTo)));
            PropertyChanged(this, new PropertyChangedEventArgs(nameof(WindowTop)));
            PropertyChanged(this, new PropertyChangedEventArgs(nameof(WindowTopTo)));
        }

This function creates the rect using the new properties of the window and then changes the values of the variables used in the XAML corresponding to the change. The end result looks like this.

Image 8

Balloons

Once the game finishes, balloons fly from the ground. To make this effect, several new methods were used.

Thread CreateBalloons = new Thread(BalloonRunner);
CreateBalloons.Start();


        public void BalloonRunner()
        {
            sbkGameEngine.BalloonRun = true;

            while (sbkGameEngine.BalloonRun)
            {
                if (Balloons)
                { 
                    Dispatcher.Invoke(() =>
                    CreateBalloon()
                    );
                }
                Dispatcher.Invoke(() =>
                    MoveBalloons()
                );
                Thread.SpinWait(1000000);
            }
        }

This thread calls the BalloonRunner method, which repeatedly calls CreateBalloon and MoveBalloon. However, it only calls CreateBalloon if the Balloon boolean is set to true, which is only done when the game finishes.

Interval -= 10;
if (Interval < 1)
{
...
Interval = RandomInt.Next(90, 150);
}

The above code uses the int Interval to only run the code every couple of calls. This allows for the balloons not to fill up the screen too quickly, and to take their time.

            if (NumberOfBalloons == 99)
            {
                Balloons = false;
            }

This code checks if the variable NumberOfBalloons is over 99, and if so, will end the thread by disabling the runner. NumberOfBalloons goes up by one every time a balloon is created.

int BalloonColor = RandomInt.Next(1, 6);
switch (BalloonColor)
{
    case 1:
        BalloonImage.ImageSource = new BitmapImage(new Uri("pack://application:,,,/PongsSBK;component/pictures/RedBalloon.png"));
        break;

    case 2:
        BalloonImage.ImageSource = new BitmapImage(new Uri("pack://application:,,,/PongsSBK;component/pictures/OrangeBalloon.png"));
        break;

    case 3:
        BalloonImage.ImageSource = new BitmapImage(new Uri("pack://application:,,,/PongsSBK;component/pictures/YellowBalloon.png"));
        break;

    case 4:
        BalloonImage.ImageSource = new BitmapImage(new Uri("pack://application:,,,/PongsSBK;component/pictures/GreenBalloon.png"));
        break;

    case 5:
        BalloonImage.ImageSource = new BitmapImage(new Uri("pack://application:,,,/PongsSBK;component/pictures/BlueBalloon.png"));
        break;
}

The code uses the switch function to replace 5 if blocks. RandomInt creates a random integer from 1-5 and then the switch function checks which number it got and adds the corresponding balloon color to the brush BalloonColor.

 Rectangle NewBalloon = new Rectangle
 {
     Tag = "Balloon",
     Width = 37,
     Height = 47,
     Fill = BalloonImage
 };

 Canvas.SetTop(NewBalloon, (int)((System.Windows.Controls.Panel)Application.Current.MainWindow.Content).ActualHeight);
 Canvas.SetLeft(NewBalloon, RandomInt.Next(0, (int)((System.Windows.Controls.Panel)Application.Current.MainWindow.Content).ActualWidth) - (NewBalloon.Width / 2));
 Board.Children.Add(NewBalloon);

This code creates a rectangles using a fill of the balloon brush from before. This allows for the rectangle to display the image of the balloon, creating a balloon.

private void MoveBalloons()
{
    foreach (var x in Board.Children.OfType<Rectangle>())
    {
        if ((string)x.Tag == "Balloon")
        {
            Canvas.SetTop(x, Canvas.GetTop(x) - speed);
            Canvas.SetLeft(x, Canvas.GetLeft(x) - (1 * RandomInt.Next(-1, 2)));
        }

        if (Canvas.GetTop(x) < 0 - x.Height)
        {
            itemRemover.Add(x);
        }
    }
    foreach (Rectangle x in itemRemover)
    {
        Board.Children.Remove(x);
    }
}

The MoveBalloons methord uses the foreach function to move each balloon. It first checks every rectangle if it has the tag "balloon", which only the balloons have, and then it moves it up and randomly to the left. Once a balloon surpasses a height, it's added to list itemRemover. The next foreach function check every rectnagle if it's on top of the window, and deletes if it does.

Doing all off this creates balloons that fly when the game completes. The game completes when a person gets a certain amount of points, defaulted to 5.  

Running the Application

  1. Go to the GitHub link: click here
  2. Go to the folder labeled "Quick Demo"
  3. Download the "ZippedDemo.zip"
  4. Unzip the file
  5. Run Pongs.exe

History

v1.0 -- May 5th 2024 - First version

v1.1 -- May 22nd 2024 - Revised abstract

v1.2 -- May 23rd 2024 - Updated steps to run the application

v1.3 -- May 24th 2024 - Updated code

v1.4 -- May 27 2024 - Added settings header

v1.5 -- May 29 2024 - Added figure numbers and descriptions

v1.6 -- May 31 2024 - Fixed gif that was previously not loading

v2.0 -- June 2 2024 - Added color pickers to settings and updated code, settings, and code explanations.

v2.1 -- June 9 2024 - Made code sections more concise

v2.2 -- June 16 2024 - Added screen shake and added more code explanations.

v2.3 -- July 19 2024 - Added variables to replace GetLeft and GetTop functions

References

1. https://stackoverflow.com/

2. https://learn.microsoft.com/en-us/dotnet/ 

 

If you found this article helpful / interesting, please don't forget to Vote! 

License

This article, along with any associated source code and files, is licensed under The MIT License


Written By
United States United States
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions

 
QuestionA Couple of Suggestions Pin
George Swan21-Jun-24 22:55
mveGeorge Swan21-Jun-24 22:55 

Thanks for the interesting and well-written piece. It’s obvious that you have worked hard on this project so I’d like, if I may, to offer a couple of simple tips that I’ve found useful in reducing the amount of time and effort needed to produce robust and easily managed code.

First off, adopt the Single-responsibility principle. Methods should do one thing only, be self-contained and easy to debug. Which means that methods are short, have a single pathway and are devoid of branching if statements. Use dependency injection and code to interfaces rather than concrete types to facilitate debugging.


Secondly, do not repeat yourself. This maxim applies not only to verbatim code duplication but also to treating related entities as individuals rather than members of the same class or interface type. For example, in the game, you have, in effect, three related game entities LeftPaddle, RightPaddle and Ball. Each has 4 position coordinates. I would use a single class GameEntity: IGameEntity with the coordinates defined within it and an UpdatePosition method that’s used to update them.



I could go on but nobody likes a smart alec so I’ll stop there. Best wishes, George.


AnswerRe: A Couple of Suggestions Pin
Sbk12324-Jun-24 8:47
Sbk12324-Jun-24 8:47 
GeneralMy vote of 5 Pin
jazim12-Jun-24 1:10
professionaljazim12-Jun-24 1:10 
GeneralRe: My vote of 5 Pin
Sbk12312-Jun-24 11:49
Sbk12312-Jun-24 11:49 
QuestionComputer Player Pin
Hyland Computer Systems24-May-24 4:09
Hyland Computer Systems24-May-24 4:09 
AnswerRe: Computer Player Pin
Sbk12324-May-24 9:35
Sbk12324-May-24 9:35 
GeneralRe: Computer Player Pin
hulinning212-Jun-24 2:19
hulinning212-Jun-24 2:19 
GeneralMy vote of 5 Pin
theDiscountCodes24-May-24 1:42
professionaltheDiscountCodes24-May-24 1:42 
GeneralRe: My vote of 5 Pin
Sbk12324-May-24 9:24
Sbk12324-May-24 9:24 
GeneralMy vote of 5 Pin
Ștefan-Mihai MOGA22-May-24 21:12
professionalȘtefan-Mihai MOGA22-May-24 21:12 
GeneralRe: My vote of 5 Pin
Sbk12323-May-24 9:42
Sbk12323-May-24 9:42 
QuestionMy vote of 5 Pin
Peter Huber SG19-May-24 23:58
mvaPeter Huber SG19-May-24 23:58 
AnswerRe: My vote of 5 Pin
Sbk12320-May-24 12:35
Sbk12320-May-24 12:35 
GeneralRe: My vote of 5 Pin
Peter Huber SG20-May-24 15:45
mvaPeter Huber SG20-May-24 15:45 
GeneralRe: My vote of 5 Pin
Sbk12322-May-24 10:59
Sbk12322-May-24 10:59 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.