mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-20 19:09:23 +02:00
173 lines
5.6 KiB
ObjectPascal
173 lines
5.6 KiB
ObjectPascal
{*****************************************************************************}
|
|
{
|
|
$Id$
|
|
This file is part of the Free Pascal's "Free Components Library".
|
|
Copyright (c) 2003 by Mazen NEIFER of the Free Pascal development team
|
|
|
|
PNM writer implementation.
|
|
|
|
See the file COPYING.FPC, included in this distribution,
|
|
for details about the copyright.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
}
|
|
{*****************************************************************************}
|
|
{Support for writing PNM (Portable aNyMap) formats added :
|
|
* PBM (P1,P4) : Portable BitMap format : 1 bit per pixel
|
|
* PGM (P2,P5) : Portable GrayMap format : 8 bits per pixel
|
|
* PPM (P5,P6) : Portable PixelMap foramt : 24 bits per pixel}
|
|
{$mode objfpc}{$h+}
|
|
unit FPWritePNM;
|
|
interface
|
|
|
|
uses FPImage, classes, sysutils;
|
|
|
|
type
|
|
|
|
TFPWriterPNM = class(TFPCustomImageWriter)
|
|
private
|
|
BitMapType:Integer;
|
|
protected
|
|
procedure InternalWrite(Stream:TStream;Img:TFPCustomImage);override;
|
|
public
|
|
constructor Create(aBitMapType:Integer);
|
|
end;
|
|
|
|
implementation
|
|
|
|
constructor TFPWriterPNM.Create(aBitMapType:Integer);
|
|
begin
|
|
inherited Create;
|
|
BitMapType:=aBitMapType;
|
|
end;
|
|
procedure TFPWriterPNM.InternalWrite(Stream:TStream;Img:TFPCustomImage);
|
|
function SaveHeader(stream:TStream):boolean;
|
|
const
|
|
MagicWords:Array[1..6]OF String[2]=('P1','P2','P3','P4','P5','P6');
|
|
var
|
|
PNMInfo:String;
|
|
strWidth,StrHeight:String[15];
|
|
begin
|
|
SaveHeader:=false;
|
|
with Img do
|
|
begin
|
|
Str(Img.Width,StrWidth);
|
|
Str(Img.Height,StrHeight);
|
|
end;
|
|
PNMInfo:=Concat(MagicWords[BitMapType],#10,StrWidth,#32,StrHeight,#10);
|
|
if BitMapType in [2,3,5,6]
|
|
then
|
|
PNMInfo:=Concat(PNMInfo,'255'#10);
|
|
stream.seek(0,soFromBeginning);
|
|
stream.Write(PNMInfo[1],Length(PNMInfo));
|
|
SaveHeader := true;
|
|
end;
|
|
var
|
|
Row,Coulumn,nBpLine,i:Integer;
|
|
aColor:TFPColor;
|
|
aLine:PByte;
|
|
strCol:String[3];
|
|
begin
|
|
SaveHeader(Stream);
|
|
case BitMapType of
|
|
1:nBpLine:=Img.Width*2;{p p p}
|
|
2:nBpLine:=Img.Width*4;{lll lll lll}
|
|
3:nBpLine:=Img.Width*3*4;{rrr ggg bbb rrr ggg bbb}
|
|
4:begin
|
|
nBpLine:=Img.Width SHR 3;
|
|
if(Img.Width AND $0F)<>0
|
|
then
|
|
Inc(nBpLine);
|
|
end;
|
|
5:nBpLine:=Img.Width;
|
|
6:nBpLine:=Img.Width*3;
|
|
end;
|
|
GetMem(aLine,nBpLine);//3 extra byte for BMP 4Bytes alignement.
|
|
for Row:=0 to img.Height-1 do
|
|
begin
|
|
FillChar(aLine^,nBpLine,0);
|
|
for Coulumn:=0 to img.Width-1 do
|
|
begin
|
|
aColor:=img.Colors[Coulumn,Row];
|
|
with aColor do
|
|
case BitMapType of
|
|
1:begin
|
|
if(Red<=$2F00)or(Green<=$2F00)or(Blue<=$2F00)
|
|
then
|
|
aLine[2*Coulumn]:=Ord('1')
|
|
else
|
|
aLine[2*Coulumn]:=Ord('0');
|
|
aLine[2*Coulumn+1]:=32;
|
|
end;
|
|
2:begin
|
|
Str(Hi(Word(Round(Red*0.299+Green*0.587+Blue*0.114))),strCol);
|
|
for i:=0 to Length(StrCol)-1 do
|
|
aLine[4*Coulumn+i]:=Ord(StrCol[i+1]);
|
|
for i:=Length(StrCol) to 4 do
|
|
aLine[4*Coulumn+i]:=32;
|
|
end;
|
|
3:begin
|
|
Str(Hi(Red),strCol);
|
|
for i:=0 to Length(StrCol)-1 do
|
|
aLine[4*(3*Coulumn)+i]:=Ord(StrCol[i+1]);
|
|
for i:=Length(StrCol) to 4 do
|
|
aLine[4*(3*Coulumn)+i]:=32;
|
|
Str(Hi(Green),strCol);
|
|
for i:=0 to Length(StrCol)-1 do
|
|
aLine[4*(3*Coulumn+1)+i]:=Ord(StrCol[i+1]);
|
|
for i:=Length(StrCol) to 4 do
|
|
aLine[4*(3*Coulumn+1)+i]:=32;
|
|
Str(Hi(Blue),strCol);
|
|
for i:=0 to Length(StrCol)-1 do
|
|
aLine[4*(3*Coulumn+2)+i]:=Ord(StrCol[i+1]);
|
|
for i:=Length(StrCol) to 4 do
|
|
aLine[4*(3*Coulumn+2)+i]:=32;
|
|
end;
|
|
4:if(Red<=$2F00)or(Green<=$2F00)or(Blue<=$2F00)
|
|
then
|
|
aLine[Coulumn shr 3]:=aLine[Coulumn shr 3] or ($80 shr (Coulumn and $07));
|
|
5:aLine[Coulumn]:=Hi(Word(Round(Red*0.299+Green*0.587+Blue*0.114)));
|
|
6:begin
|
|
aLine[3*Coulumn]:=Hi(Red);
|
|
aLine[3*Coulumn+1]:=Hi(Green);
|
|
aLine[3*Coulumn+2]:=Hi(Blue);
|
|
end;
|
|
end;
|
|
end;
|
|
Stream.Write(aLine^,nBpLine);
|
|
end;
|
|
FreeMem(aLine,nBpLine);
|
|
end;
|
|
|
|
initialization
|
|
ImageHandlers.RegisterImageWriter ('PBM Format', 'pbm', TFPWriterPNM);
|
|
end.
|
|
{
|
|
$Log$
|
|
Revision 1.1 2003-09-30 06:23:32 mazen
|
|
+ Support for PNM (Portable aNyMap) formats
|
|
|
|
Revision 1.5 2003/09/09 11:28:23 mazen
|
|
* fixing copyright section in the file header
|
|
|
|
Revision 1.4 2003/09/08 14:08:48 mazen
|
|
- all common defintions are now included into bmpcomn unit
|
|
- removed erronous code (causing exception)
|
|
|
|
Revision 1.3 2003/09/08 10:38:56 luk
|
|
- removed debug info
|
|
* prevented exceptions when using non indexed images
|
|
|
|
Revision 1.2 2003/09/04 22:29:43 luk
|
|
* correct color conversion (prevent range check errors)
|
|
|
|
Revision 1.1 2003/09/04 12:02:21 mazen
|
|
+ fpwritebmp.pas renamed to fpwritebmp.pp
|
|
|
|
Revision 1.1 2003/09/04 08:44:32 mazen
|
|
+ Adds support of writing BMP files
|
|
|
|
}
|