C# 图片加水印例程

using System;
using System.IO;
using System.Collections;
using System.Drawing;
using System.Drawing.Drawing2D;
using System.Drawing.Imaging;

namespace Imag_writer
{
/// <summary>
/// 水印的类型
/// </summary>
public enum WaterMarkType
{
   /// <summary>
   /// 文字水印
   /// </summary>
   TextMark,
   /// <summary>
   /// 图片水印
   /// </summary>
   //ImageMark // 暂时只能添加文字水印
};

/// <summary>
/// 水印的位置
/// </summary>
public enum WaterMarkPosition
{
   /// <summary>
   /// 左上角
   /// </summary>
   WMP_Left_Top,
   /// <summary>
   /// 左下角
   /// </summary>
   WMP_Left_Bottom,
   /// <summary>
   /// 右上角
   /// </summary>
   WMP_Right_Top,
   /// <summary>
   /// 右下角
   /// </summary>
   WMP_Right_Bottom
};

/// <summary>
/// 处理图片的类(包括加水印,生成缩略图)
/// </summary>
public class ImageWaterMark
{
   public ImageWaterMark()
   {
    //
    // TODO: 在此处添加构造函数逻辑
    //
   }

   #region 给图片加水印
   /// <summary>
   /// 添加水印(分图片水印与文字水印两种)
   /// </summary>
   /// <param name="oldpath">原图片绝对地址</param>
   /// <param name="newpath">新图片放置的绝对地址</param>
   /// <param name="wmtType">要添加的水印的类型</param>
   /// <param name="sWaterMarkContent">水印内容,若添加文字水印,此即为要添加的文字;
   /// 若要添加图片水印,此为图片的路径</param>
   public void addWaterMark(string oldpath, string newpath,
    WaterMarkType wmtType, string sWaterMarkContent)
   {
    try
    {
     Image image = Image.FromFile(oldpath);

     Bitmap b = new Bitmap(image.Width, image.Height,
      PixelFormat.Format24bppRgb);

     Graphics g = Graphics.FromImage(b);
     g.Clear(Color.White);
     g.SmoothingMode = SmoothingMode.HighQuality;
     g.InterpolationMode = InterpolationMode.High;

     g.DrawImage(image, 0, 0, image.Width, image.Height);

     switch (wmtType)
     {
      /*case WaterMarkType.ImageMark:
       //图片水印
       this.addWatermarkImage(g,
        Page.Server.MapPath(Watermarkimgpath),
        WatermarkPosition,image.Width,image.Height);
       break;*/
      case WaterMarkType.TextMark:
       //文字水印
       this.addWatermarkText(g, sWaterMarkContent, "WM_BOTTOM_RIGHT",
        image.Width, image.Height);
       break;
     }

     b.Save(newpath);
     b.Dispose();
     image.Dispose();
    }
    catch
    {
     if(File.Exists(oldpath))
     {
      File.Delete(oldpath);
     }
    }
    finally
    {
     if(File.Exists(oldpath))
     {
      File.Delete(oldpath);
     }
    }  
   }

   /// <summary>
   ///   加水印文字
   /// </summary>
   /// <param name="picture">imge 对象</param>
   /// <param name="_watermarkText">水印文字内容</param>
   /// <param name="_watermarkPosition">水印位置</param>
   /// <param name="_width">被加水印图片的宽</param>
   /// <param name="_height">被加水印图片的高</param>
   ******* void addWatermarkText(Graphics picture, string _watermarkText,
    string _watermarkPosition, int _width, int _height)
   {
    // 确定水印文字的字体大小
    int[] sizes = new int[]{32, 30, 28, 26, 24, 22, 20, 18, 16, 14, 12, 10, 8, 6, 4};
    Font crFont = null;
    SizeF crSize = new SizeF();
    for (int i = 0;i < sizes.Length; i++)
    {
     crFont = new Font("Arial Black", sizes[i], FontStyle.Bold);
     crSize = picture.MeasureString(_watermarkText, crFont);

     if((ushort)crSize.Width < (ushort)_width)
     {
      break;
     }
    }

    // 生成水印图片(将文字写到图片中)
    Bitmap floatBmp = new Bitmap((int)crSize.Width + 3,
          (int)crSize.Height + 3, PixelFormat.Format32bppArgb);
    Graphics fg=Graphics.FromImage(floatBmp);
    PointF pt=new PointF(0,0);

    // 画阴影文字
    Brush TransparentBrush0 = new SolidBrush(Color.FromArgb(255, Color.Black));
    Brush TransparentBrush1 = new SolidBrush(Color.FromArgb(255, Color.Black));
    fg.DrawString(_watermarkText,crFont,TransparentBrush0, pt.X, pt.Y + 1);
    fg.DrawString(_watermarkText,crFont,TransparentBrush0, pt.X + 1, pt.Y);

    fg.DrawString(_watermarkText,crFont,TransparentBrush1, pt.X + 1, pt.Y + 1);
    fg.DrawString(_watermarkText,crFont,TransparentBrush1, pt.X, pt.Y + 2);
    fg.DrawString(_watermarkText,crFont,TransparentBrush1, pt.X + 2, pt.Y);

    TransparentBrush0.Dispose();
    TransparentBrush1.Dispose();

    // 画文字
    fg.SmoothingMode=System.Drawing.Drawing2D.SmoothingMode.HighQuality;
    fg.DrawString(_watermarkText,
     crFont, new SolidBrush(Color.White),
     pt.X, pt.Y, StringFormat.GenericDefault);

    // 保存刚才的操作
    fg.Save();
    fg.Dispose();
    // floatBmp.Save("d:\\WebSite\\DIGITALKM\\ttt.jpg");

    // 将水印图片加到原图中
    this.addWatermarkImage(
     picture,
     new Bitmap(floatBmp),
     "WM_BOTTOM_RIGHT",
     _width,
     _height);
   }

   /// <summary>
   ///   加水印图片
   /// </summary>
   /// <param name="picture">imge 对象</param>
   /// <param name="iTheImage">Image对象(以此图片为水印)</param>
   /// <param name="_watermarkPosition">水印位置</param>
   /// <param name="_width">被加水印图片的宽</param>
   /// <param name="_height">被加水印图片的高</param>
   ******* void addWatermarkImage( Graphics picture,Image iTheImage,
    string _watermarkPosition,int _width,int _height)
   {
    Image watermark = new Bitmap(iTheImage);

    ImageAttributes imageAttributes = new ImageAttributes();
    ColorMap colorMap = new ColorMap();

    colorMap.OldColor = Color.FromArgb(255, 0, 255, 0);
    colorMap.NewColor = Color.FromArgb(0, 0, 0, 0);
    ColorMap[] remapTable = {colorMap};

    imageAttributes.SetRemapTable(remapTable, ColorAdjustType.Bitmap);

    float[][] colorMatrixElements = {
             new float[] {1.0f, 0.0f, 0.0f, 0.0f, 0.0f},
             new float[] {0.0f, 1.0f, 0.0f, 0.0f, 0.0f},
             new float[] {0.0f, 0.0f, 1.0f, 0.0f, 0.0f},
             new float[] {0.0f, 0.0f, 0.0f, 0.3f, 0.0f},
             new float[] {0.0f, 0.0f, 0.0f, 0.0f, 1.0f}
            };

    ColorMatrix colorMatrix = new ColorMatrix(colorMatrixElements);

    imageAttributes.SetColorMatrix(colorMatrix, ColorMatrixFlag.Default, ColorAdjustType.Bitmap);

    int xpos = 0;
    int ypos = 0;
    int WatermarkWidth = 0;
    int WatermarkHeight = 0;
    double bl = 1d;

    //计算水印图片的比率
    //取背景的1/4宽度来比较
    if ((_width > watermark.Width * 4) && (_height > watermark.Height * 4))
    {
     bl = 1;
    }
    else if ((_width > watermark.Width * 4) && (_height<watermark.Height * 4))
    {
     bl = Convert.ToDouble(_height / 4) / Convert.ToDouble(watermark.Height);
            
    }
    else if ((_width < watermark.Width * 4) && (_height > watermark.Height * 4))
    {
     bl = Convert.ToDouble(_width / 4) / Convert.ToDouble(watermark.Width);
    }
    else
    {
     if ((_width * watermark.Height) > (_height * watermark.Width))
     {
      bl = Convert.ToDouble(_height / 4) / Convert.ToDouble(watermark.Height);
                    
     }
     else
     {
      bl = Convert.ToDouble(_width / 4) / Convert.ToDouble(watermark.Width);
                    
     }
            
    }

    WatermarkWidth = Convert.ToInt32(watermark.Width * bl);
    WatermarkHeight = Convert.ToInt32(watermark.Height * bl);

    switch (_watermarkPosition)
    {
     case "WM_TOP_LEFT":
      xpos = 10;
      ypos = 10;
      break;
     case "WM_TOP_RIGHT":
      xpos = _width - WatermarkWidth - 10;
      ypos = 10;
      break;
     case "WM_BOTTOM_RIGHT":
      xpos = _width - WatermarkWidth - 10;
      ypos = _height -WatermarkHeight - 10;
      break;
     case "WM_BOTTOM_LEFT":
      xpos = 10;
      ypos = _height - WatermarkHeight - 10;
      break;
    }

    picture.DrawImage(
     watermark,
     new Rectangle(xpos, ypos, WatermarkWidth, WatermarkHeight),
     0,
     0,
     watermark.Width,
     watermark.Height,
     GraphicsUnit.Pixel,
     imageAttributes);

    watermark.Dispose();
    imageAttributes.Dispose();
   }

   /// <summary>
   ///   加水印图片
   /// </summary>
   /// <param name="picture">imge 对象</param>
   /// <param name="WaterMarkPicPath">水印图片的地址</param>
   /// <param name="_watermarkPosition">水印位置</param>
   /// <param name="_width">被加水印图片的宽</param>
   /// <param name="_height">被加水印图片的高</param>
   ******* void addWatermarkImage( Graphics picture,string WaterMarkPicPath,
    string _watermarkPosition,int _width,int _height)
   {
    Image watermark = new Bitmap(WaterMarkPicPath);

    this.addWatermarkImage(picture, watermark, _watermarkPosition, _width,
     _height);
   }
   #endregion

   #region 生成缩略图
   /// <summary>
   /// 保存图片
   /// </summary>
   /// <param name="image">Image 对象</param>
   /// <param name="savePath">保存路径</param>
   /// <param name="ici">指定格式的编解码参数</param>
   ******* void SaveImage(Image image, string savePath, ImageCodecInfo ici)
   {
    //设置 原图片 对象的 EncoderParameters 对象
    EncoderParameters parameters = new EncoderParameters(1);
    parameters.Param[0] = new EncoderParameter(
     System.Drawing.Imaging.Encoder.Quality, ((long) 90));
    image.Save(savePath, ici, parameters);
    parameters.Dispose();
   }

   /// <summary>
   /// 获取图像编码解码器的所有相关信息
   /// </summary>
   /// <param name="mimeType">包含编码解码器的多用途网际邮件扩充协议 (MIME) 类型的字符串</param>
   /// <returns>返回图像编码解码器的所有相关信息</returns>
   ******* ImageCodecInfo GetCodecInfo(string mimeType)
   {
    ImageCodecInfo[] CodecInfo = ImageCodecInfo.GetImageEncoders();
    foreach(ImageCodecInfo ici in CodecInfo)
    {
     if(ici.MimeType == mimeType)
      return ici;
    }
    return null;
   }

   /// <summary>
   /// 生成缩略图
   /// </summary>
   /// <param name="sourceImagePath">原图片路径(相对路径)</param>
   /// <param name="thumbnailImagePath">生成的缩略图路径,如果为空则保存为原图片路径(相对路径)</param>
   /// <param name="thumbnailImageWidth">缩略图的宽度(高度与按源图片比例自动生成)</param>
   public void ToThumbnailImages(
    string SourceImagePath,
    string ThumbnailImagePath,
    int ThumbnailImageWidth)
   {
    Hashtable htmimes = new Hashtable();
    htmimes[".jpeg"] = "image/jpeg";
    htmimes[".jpg"] = "image/jpeg";   
    htmimes[".png"] = "image/png";   
    htmimes[".tif"] = "image/tiff";
    htmimes[".tiff"] = "image/tiff";
    htmimes[".bmp"] = "image/bmp";
    htmimes[".gif"] = "image/gif";

    // 取得原图片的后缀
    string sExt = SourceImagePath.Substring(
     SourceImagePath.LastIndexOf(".")).ToLower();

    //从 原图片 创建 Image 对象
    Image image = Image.FromFile(SourceImagePath);  
    int num = ((ThumbnailImageWidth / 4) * 3);
    int width = image.Width;
    int height = image.Height;

    //计算图片的比例
    if ((((double) width) / ((double) height)) >= 1.3333333333333333f)
    {
     num = ((height * ThumbnailImageWidth) / width);
    }
    else
    {
     ThumbnailImageWidth = ((width * num) / height);
    }
    if ((ThumbnailImageWidth < 1) || (num < 1))
    {
     return;
    }

    //用指定的大小和格式初始化 Bitmap 类的新实例
    Bitmap bitmap = new Bitmap(ThumbnailImageWidth, num,
     PixelFormat.Format32bppArgb);

    //从指定的 Image 对象创建新 Graphics 对象
    Graphics graphics = Graphics.FromImage(bitmap);

    //清除整个绘图面并以透明背景色填充
    graphics.Clear(Color.Transparent);

    graphics.SmoothingMode = SmoothingMode.HighQuality;
    graphics.InterpolationMode = InterpolationMode.High;

    //在指定位置并且按指定大小绘制 原图片 对象
    graphics.DrawImage(image, new Rectangle(0, 0, ThumbnailImageWidth, num));
    image.Dispose();
   
    try
    {   
     //将此 原图片 以指定格式并用指定的编解码参数保存到指定文件
     SaveImage(bitmap, ThumbnailImagePath,
      GetCodecInfo((string)htmimes[sExt]));
    }
    catch(System.Exception e)
    {
     throw e;
    }
   }
   #endregion
}
}

 

转载于:https://www.cnblogs.com/hackpig/archive/2010/02/15/1668379.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/299471.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

从N个元素中选择第i小的元素

时常在笔试,面试题中看到这个问题,《算法导论》中给出了很好的解答。 Selection of the ith smallest element of the array A can be done in θ(n) times. The psuedocode is following: CodeRandomized_Select(A,p,r,i){ if pr then return A[p] qRandomized_…

Blazor 模板化组件开发指南

翻译自 Waqas Anwar 2021年4月15日的文章 《A Developer’s Guide To Blazor Templated Components》 [1]在我之前的一篇文章 Blazor 组件入门指南中&#xff0c;我介绍了组件参数&#xff0c;并向您展示了如何将数据作为参数传递给 Blazor 组件以定制化其功能。在这篇文章中&a…

别太贪婪,这些技能能让你一辈子满足

全世界只有3.14 % 的人关注了青少年数学之旅在这个资讯丰富且易获取的时代&#xff0c;越来越多的人不愿意花时间阅读书籍&#xff0c;碎片化阅读成了主流。人们获取的东西多而杂&#xff0c;很难系统、全面。海量信息对人是冲击&#xff0c;更是诱惑。谁不想了解天下奇闻&…

纳尼???我JVM优化过头了,直接把异常信息优化没了?怎么办

你好呀&#xff0c;我是why。 你猜这次我又要写个啥没有卵用的知识点呢&#xff1f; 不好意思&#xff0c;问的稍微有点早了&#xff0c;啥提示都没给&#xff0c;咋猜呢&#xff0c;对吧&#xff1f; 先给你上个代码&#xff1a; public class ExceptionTest {public stati…

angular $watch

在scope内置的所有函数中&#xff0c;用得最多的可能就是$watch 函数了&#xff0c;当你的数据模型中某一部分发生变化时&#xff0c;$watch函数可以向你发出通知。你可以监控单个对象的属性&#xff0c;也可以监控需要经过计算的结果&#xff08;函数&#xff09;&#xff0c;…

C# 读写ACCESS的OLE对象,演示图片与长文件的读写

网络上的读写OLE对象的代码是多&#xff0c;不过多是转载的&#xff0c;大部分人从来都没实际测试过&#xff0c;只是COPY来COPY去。我重来没看到一个真正可以运行的东东。没办法&#xff0c;只有自力更生&#xff0c;花了一点时间出了点研究成果&#xff0c;写到这里做个记录。…

RHCE课程-初级部分6、编辑工具VIM,网络配置,进程优先,日志文件简介。

我们通常用各种编辑工具来处理文本文件 常用的编辑工具:VIVIMEMACSVI和VIM的区别它们都是多模式编辑器&#xff0c;不同的是vim 是vi的升级版本&#xff0c;它不仅兼容vi的所有指令&#xff0c;而且还有一些新的特性在里面。vim的这些优势主要体现在以下几个方面&#xff1a;易…

WPF等待动画

WPF开发者QQ群&#xff1a; 340500857 | 微信群 -> 进入公众号主页 加入组织欢迎转发、分享、点赞、在看&#xff0c;谢谢~。 01—效果预览效果预览&#xff08;更多效果请下载源码体验&#xff09;&#xff1a;02—代码如下一、CycleLoading.cs 代码如下using System.Win…

假期别在家里要发霉了?可以靠他们度过无聊时光

全世界只有3.14 % 的人关注了青少年数学之旅在这个资讯丰富且易获取的时代&#xff0c;越来越多的人不愿意花时间阅读书籍&#xff0c;碎片化阅读成了主流。人们获取的东西多而杂&#xff0c;很难系统、全面。海量信息对人是冲击&#xff0c;更是诱惑。谁不想了解天下奇闻&…

cs6序列号 mac版photoshop_重磅!Parallels Desktop 16 M1版发布

Parallels 现已发布 Parallels Desktop 16 技术预览版&#xff0c;可以在搭载 M1 芯片的 Mac 电脑上运行。该公司表示&#xff0c;它创建了一个 “使用 Apple M1 Mac 芯片硬件辅助虚拟化的新虚拟化引擎”&#xff0c;允许用户在虚拟机中运行基于 Arm 的操作系统&#xff0c;例如…

华为二面!!!面试官直接问我Java中到底什么是NIO?这不是直接送分题???

华为二面&#xff01;&#xff01;&#xff01;面试官直接问我Java中到底什么是NIO&#xff1f;这不是直接送分题&#xff1f;&#xff1f;&#xff1f;什么是NIO缓冲区(Buffer)缓冲区类型获取缓冲区核心属性核心方法非直接缓冲区和直接缓冲区非直接缓冲区直接缓冲区通道(Chann…

REST

REST 定义了一组体系架构原则&#xff0c;您可以根据这些原则设计以系统资源为中心的 Web 服务REST 对 Web 的影响非常大&#xff0c;由于其使用相当方便&#xff0c;已经普遍地取代了基于 SOAP 和 WSDL 的接口设计。REST Web 服务&#xff0c;其具体实现应该遵循四个基本设计原…

Delphi中的容器类(3)

TBucketList和TObjectBucketList类 从Delphi6开始&#xff0c;VCL的Contnrs单元中又增加了两个新的容器类TBucketList和TObjectBucketList。TBucketList实际上也是一个简单基于哈希表的指针-指针对列表。接口定义如下&#xff1a; TBucketList class(TCustomBucketList)… pu…

修改PATH导致fedora无法登录XWindow

用过fedora的都了解到&#xff0c;如果要使用像ifconfig,arp等这些外部命令&#xff0c;就必需加入路径&#xff0c;有点麻烦&#xff08;我懒人一个-_-&#xff09;。所以我就在/etc/profile文件结尾处加入了一行&#xff1a;export PATH$PATH:/sbin很好&#xff0c;感觉上没问…

一文读懂哈希和一致性哈希算法

哈希 Hash 算法介绍哈希算法也叫散列算法, 不过英文单词都是 Hash, 简单一句话概括, 就是可以把任意长度的输入信息通过算法变换成固定长度的输出信息, 输出信息也就是哈希值, 通常哈希值的格式是16进制或者是10进制, 比如下面的使用 md5 哈希算法的示例md5("123456"…

延迟开学?这些教育读书公众号可以帮助孩子学习! 你都关注了吗?

全世界只有3.14 % 的人关注了青少年数学之旅受新型冠状病毒疫情影响&#xff0c;日前&#xff0c;教育部印发《关于2020年春季学期延期开学的通知》。推迟开学时间&#xff0c;意味着寒假的延长。为此&#xff0c;小编精选这些优质的教育号和读书号帮助孩子学习&#xff01;理想…

go方法的深入探究(7.21增补)

2019独角兽企业重金招聘Python工程师标准>>> 1&#xff09;哪些类型可以有方法&#xff1a; 1&#xff09;只能对命名类型和命名类型的指针编写方法&#xff1b; 2&#xff09;不能对接口类型和接口类型的指针编写方法&#xff1b; 3&#xff09;只能在定义命名类型…

element文件上传有文件但是后台接收为空_程序员提高篇:大规格文件(G)是如何做分片优化的?...

作者&#xff1a;凹凸实验室 链接&#xff1a;https://juejin.im/post/5ebb4346e51d451ef53793ad整体思路第一步是结合项目背景&#xff0c;调研比较优化的解决方案。 文件上传失败是老生常谈的问题&#xff0c;常用方案是将一个大文件切片成多个小文件&#xff0c;并行请求接口…

你连简单的枚举类都不知道,还敢说自己会Java???滚出我的公司

枚举类型是Java 5中新增的特性&#xff0c;它是一种特殊的数据类型&#xff0c;之所以特殊是因为它既是一种类(class)类型却又比类类型多了些特殊的约束&#xff0c;但是这些约束的存在也造就了枚举类型的简洁性、安全性以及便捷性。当需要定义一组常量时&#xff0c;强烈建议使…

爱情第七课,被爱的秘密

我们都想要被爱。 这和懒着&#xff0c;却想有好东西吃时的心态一样。但爱的需要却和生理需要不同&#xff0c;它是一种更高级的精神需要。 在小时候&#xff0c;我们还不懂得什么是爱时&#xff0c;就接收到父母爱的信号&#xff1a;被照顾&#xff0c;被养育&#xff0c;被善…