4

是否可以以灰度绘制任何表单(不覆盖 Paint 方法)。

如果我在 Modal() 对话框中显示表单,我不会将其父级显示为灰度。我在 Visual Studio 扩展管理器中注意到了这一点。如果进度条正在下载包,则底层窗口将显示为灰色。

我在想这个:

private void Button1_Click(object sender, EventArgs e)
{
    using (var dialog = new Form2())
    {
        SetGrayscale(this, true);
        dialog.ShowDialog();
        SetGrayscale(this, false);
    }
}

更新

只是设置Form.Enabled = false;不是我想要的。这看起来不如我的表单的灰度表示。我认为compizLinux 的窗口装饰器使用无响应的应用程序做到了这一点。

4

3 回答 3

2

正如已经说过的那样,这样做的方法是在现有表单的顶部覆盖另一个控件/表单,并让它在顶部呈现这个的灰度版本,您可以使用恰好放置在原始表单上的附加表单来执行此操作, 或使用类似Panel定位在所有其他控件之上的东西。

这是一个工作示例,说明在将另一个表单完全放在第一个表单的客户区域上时如何执行此操作。如何使用它

using (Grayscale(this))
{
    MessageBox.Show("Test");
}

执行

public static Form Grayscale(Form tocover)
{
    var frm = new Form
        {
            FormBorderStyle = FormBorderStyle.None,
            ControlBox = false,
            ShowInTaskbar = false,
            StartPosition = FormStartPosition.Manual,
            AutoScaleMode = AutoScaleMode.None,
            Location = tocover.PointToScreen(tocover.ClientRectangle.Location),
            Size = tocover.ClientSize
        };
    frm.Paint += (sender, args) =>
        {
            var bmp = GetFormImageWithoutBorders(tocover);
            bmp = ConvertToGrayscale(bmp);
            args.Graphics.DrawImage(bmp, args.ClipRectangle.Location);
        };

    frm.Show(tocover);
    return frm;
}

private static Bitmap ConvertToGrayscale(Bitmap source)
{
    var bm = new Bitmap(source.Width, source.Height);
    for (int y = 0; y < bm.Height; y++)
    {
        for (int x = 0; x < bm.Width; x++)
        {
            Color c = source.GetPixel(x, y);
            var luma = (int)(c.R * 0.3 + c.G * 0.59 + c.B * 0.11);
            bm.SetPixel(x, y, Color.FromArgb(luma, luma, luma));
        }
    }
    return bm;
}

private static Bitmap GetControlImage(Control ctl)
{
    var bm = new Bitmap(ctl.Width, ctl.Height);
    ctl.DrawToBitmap(bm, new Rectangle(0, 0, ctl.Width, ctl.Height));
    return bm;
}

private static Bitmap GetFormImageWithoutBorders(Form frm)
{
    // Get the form's whole image.
    using (Bitmap wholeForm = GetControlImage(frm))
    {
        // See how far the form's upper left corner is
        // from the upper left corner of its client area.
        Point origin = frm.PointToScreen(new Point(0, 0));
        int dx = origin.X - frm.Left;
        int dy = origin.Y - frm.Top;

        // Copy the client area into a new Bitmap.
        int wid = frm.ClientSize.Width;
        int hgt = frm.ClientSize.Height;
        var bm = new Bitmap(wid, hgt);
        using (Graphics gr = Graphics.FromImage(bm))
        {
            gr.DrawImage(wholeForm, 0, 0,
                new Rectangle(dx, dy, wid, hgt),
                GraphicsUnit.Pixel);
        }
        return bm;
    }
}

注意:

  • 的实现Paint相当糟糕——实际上它应该使用双缓冲,以便将灰度图像预渲染到缓冲的图形上下文中,因此 Paint 方法只需要绘制预绘制的缓冲区内容。请参阅C# 中的自定义绘图控件 - 手动双缓冲
  • ConvertToGrayscale有点慢,但可能会加快
  • 如果有人出于任何原因设法移动原始表格,事情就会出错
  • 图像是静态的,如果基础控件被重绘,那么理想情况下顶部表单也应该重绘。我不确定如何最好地检测另一个表单的一部分何时失效。

如果我有时间我会尝试解决其中的一些问题,但以上至少给了你一个大致的想法。

请注意,在 WPF 中,这会容易得多。

资料来源:

于 2012-06-14T12:52:45.197 回答
1

我不认为有办法直接做到这一点——我认为所有的表格都是用 sRGB 渲染的。一个 hacky 的方法可能是用它的副本作为图像覆盖表单(这很容易使用Control.DrawToBitMap),然后通过一个简单的 GDI 矩阵将其传递给https://web.archive.org/web /20141230145627/http://bobpowell.net/grayscale.aspx

于 2012-06-14T12:18:45.580 回答
0

尝试这样的方法,它适用于大多数简单的控件(您需要递归到容器中才能正确切换所有控件)。

    private void button1_Click(object sender, EventArgs e)
    {
        using (var dialog = new Form())
        {
            Dictionary<Control, Tuple<Color, Color>> oldcolors = new Dictionary<Control, Tuple<Color, Color>>();
            foreach (Control ctl in this.Controls)
            {
                oldcolors.Add(ctl, Tuple.Create(ctl.BackColor, ctl.ForeColor));
                // get rough avg intensity of color
                int bg = (ctl.BackColor.R + ctl.BackColor.G + ctl.BackColor.B) / 3;
                int fg = (ctl.ForeColor.R + ctl.ForeColor.G + ctl.ForeColor.B) / 3;
                ctl.BackColor = Color.FromArgb(bg, bg, bg);
                ctl.ForeColor = Color.FromArgb(fg, fg, fg);
            }

            dialog.ShowDialog();

            foreach (Control ctl in this.Controls)
            {
                ctl.BackColor = oldcolors[ctl].Item1;
                ctl.ForeColor = oldcolors[ctl].Item2;
            }
        }
    }
于 2012-06-14T12:02:24.157 回答