web-dev-qa-db-ja.com

C#WinFormsによる透明な画像

私はVS 2008でWindowsフォームアプリケーションに取り組んでいて、1つの画像を別の画像の上に重ねて表示したいと思います。上部の画像はgifまたは透明な部分のあるものです。

基本的に私は大きな画像を持っていますが、もしそうなら小さな画像を上に置きたいので、それらはユーザーに1つの画像として表示されます。

私はピクチャーボックスを使用しようとしていますが、これは機能していないようです、何か提案はありますか?

23
Fiona

私は数日前に同じような状況にありました。画像をホストする透明なコントロールを作成できます。

using System;
using System.Windows.Forms;
using System.Drawing;

public class TransparentControl : Control
{
    private readonly Timer refresher;
    private Image _image;

    public TransparentControl()
    {
        SetStyle(ControlStyles.SupportsTransparentBackColor, true);
        BackColor = Color.Transparent;
        refresher = new Timer();
        refresher.Tick += TimerOnTick;
        refresher.Interval = 50;
        refresher.Enabled = true;
        refresher.Start();
    }

    protected override CreateParams CreateParams
    {
        get
        {
            CreateParams cp = base.CreateParams;
            cp.ExStyle |= 0x20;
            return cp;
        }
    }

    protected override void OnMove(EventArgs e)
    {
        RecreateHandle();
    }


    protected override void OnPaint(PaintEventArgs e)
    {
        if (_image != null)
        {
            e.Graphics.DrawImage(_image, (Width / 2) - (_image.Width / 2), (Height / 2) - (_image.Height / 2));
        }
    }

    protected override void OnPaintBackground(PaintEventArgs e)
    {
       //Do not Paint background
    }

    //Hack
    public void Redraw()
    {
        RecreateHandle();
    }

    private void TimerOnTick(object source, EventArgs e)
    {
        RecreateHandle();
        refresher.Stop();
    }

    public Image Image
    {
        get
        {
            return _image;
        }
        set
        {
            _image = value;
            RecreateHandle();
        }
    }
}
26
Leon Tayson

PictureBoxには、BackgroundImageとImageの2層の画像があります。これらは、描画やクリアなど、互いに独立して使用できます。

6
Nav

大きい画像/下の画像をPictureBoxに配置し、OnPaintイベントにハンドラーを追加して、e.Graphics.DrawImage()オーバーロードの1つを使用します。 Image.FromFile()を使用してイメージをロードできます。

小さい/上の画像は、オーバーレイが機能するために、アルファチャネルを持ち、背景で透明でなければなりません。 Photoshopなどでこれをかなり簡単に確認できるはずです。 PNGなど、アルファチャネルをサポートする形式で保存してください。

4
Jon Grant

Vb.netコード(Leon Taysonへのすべてのクレジット):

Imports System
Imports System.Windows.Forms
Imports System.Drawing

Public Class TransparentControl
    Inherits Control

    Private ReadOnly Local_Timer As Timer
    Private Local_Image As Image

    Public Sub New()
        SetStyle(ControlStyles.SupportsTransparentBackColor, True)
        BackColor = Color.Transparent
        Local_Timer = New Timer
        With Local_Timer
            .Interval = 50
            .Enabled = True
            .Start()
        End With

        AddHandler Local_Timer.Tick, AddressOf TimerOnClick

    End Sub

    Protected Overrides ReadOnly Property CreateParams() As System.Windows.Forms.CreateParams
        Get
            Dim cp As CreateParams
            cp = MyBase.CreateParams
            cp.ExStyle = &H20
            Return cp
        End Get
    End Property

    Protected Overrides Sub OnMove(ByVal e As System.EventArgs)
        MyBase.OnMove(e)
        RecreateHandle()
    End Sub

    Protected Overrides Sub OnPaint(ByVal e As System.Windows.Forms.PaintEventArgs)
        MyBase.OnPaint(e)

        If Local_Image IsNot Nothing Then _
            e.Graphics.DrawImage(Local_Image, New Rectangle(0, 0, (Width / 2) - (Local_Image.Width / 2), (Height / 2) - (Local_Image.Height / 2)))

    End Sub

    Protected Overrides Sub OnPaintBackground(ByVal pevent As System.Windows.Forms.PaintEventArgs)
        ' DO NOT Paint BACKGROUND
    End Sub

    ''' <summary>
    ''' Hack
    ''' </summary>
    ''' <remarks></remarks>
    Public Sub ReDraw()
        RecreateHandle()
    End Sub

    Private Sub TimerOnClick(ByVal sender As Object, ByVal e As System.EventArgs)
        RecreateHandle()
        Local_Timer.Stop()

    End Sub

    Public Property Image As Image
        Get
            Return Local_Image
        End Get
        Set(ByVal value As Image)
            Local_Image = value
            RecreateHandle()
        End Set
    End Property
End Class
3
MiBol

同様の投稿のリストは、この返信の下部で参照されます。

この返信は、pictureBoxesとWinformsに対応しています(以下の他の投稿では、WPFがすでにこれをうまく解決していることをいくつか繰り返します)

  1. Winformを作成する
  2. X2 pictureBoxes を作成します
    • foreground_pictureBox //「背景」の「前」にある画像ボックス
    • background_pictureBox //「前景」の「背後」にある画像ボックス
  3. 各pictureBox に 'Paint'イベントを追加します。
    • 「デザイナー」でオブジェクトを選択
    • 「プロパティ」タブを選択(または右クリックしてポップアップメニューから選択)
    • イベントボタンを選択します(小さな稲妻)
    • 「ペイント」イベントの右側の空のフィールドをダブルクリックします
  4. メインフォームの 'load'関数に次のコードを追加します(まだ追加されていない場合は、手順3のアプローチを使用して、 'Paint'ではなく 'on load'を選択します)。

=

private void cFeedback_Form_Load(object sender, EventArgs e)
{
    ...
    // Ensure that it is setup with transparent background
    foreground_pictureBox.BackColor = Color.Transparent;

    // Assign it's 'background'
    foreground_pictureBox.Parent = background_pictureBox;
    ...
}

5。 「background_pictureBox」の「Paint」呼び出しで:

=

private void background_pictureBox_Paint(object sender, PaintEventArgs e)
{
    ...foreground_pictureBox_Paint(sender, e);
}

6。 'foreground_pictureBox_Paint'呼び出し内で、フォアグラウンドに表示したいグラフィックス呼び出しを追加します。

このトピックは、思われるいくつかの投稿で繰り返されます:

how-to-make-picturebox-transparent

c-sharp-picturebox-transparent-background-doesnt-seem-to-work

make-overlapping-picturebox-transparent-in-c-net

a-picturebox-problem

2
S. Arseneau

単一のピクチャーボックスまたはコントロールを使用して、自分で画像を合成しなければならないことにいつも気づきました。透明なパーツを備えた2つの画像ボックスを使用することは、私にとってはうまくいきませんでした。

0
Charlie Salts