printingservices/printerdrivers/canon/CANON.CPP
author Matt Plumtree <matt.plumtree@nokia.com>
Fri, 06 Aug 2010 17:05:20 +0100
branchNewGraphicsArchitecture
changeset 143 3db46cb3f779
parent 0 5d03bc08d59c
permissions -rw-r--r--
Fix TRANSPARENCY_NONE composition for surfaces narrower than the context. Improve performance of point sample scaling with 8-bit samples, by using fixed point code Allow any non-zero value for the boolean attribute WFC_ELEMENT_SOURCE_FLIP Simplify RemoveElement code

// Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
//

#include "CANONSTD.H"

#include <banddev.h>
#include <fbs.h>
#include "CANON.H"
#include "pdrtext.h"


EXPORT_C CPrinterDevice* NewPrinterDeviceL()
	{
	CCanonDevice* device = new(ELeave) CCanonDevice;
	return device;
	}

CCanonDevice::CCanonDevice():
	CFbsDrvDevice()
	{
	__DECLARE_NAME(_S("CCanonDevice"));
	}

CCanonDevice::~CCanonDevice()
	{
	}

TInt CCanonDevice::CreateContext(CGraphicsContext*& aGC)
	{
	__ASSERT_DEBUG(iControl, Panic(ECanonControlDoesNotExist));
	CPdrControl* control = (CPdrControl*)iControl;
	return control->CreateContext(aGC);
	}

void CCanonDevice::CreateControlL(CPrinterPort* aPrinterPort)
	{
	__ASSERT_ALWAYS(aPrinterPort, Panic(ECanonRequiresPrinterPort));
	__ASSERT_ALWAYS(!iControl, Panic(ECanonControlAlreadyExists));
	__ASSERT_DEBUG(iCurrentPageSpecInTwips.iPortraitPageSize.iWidth && iCurrentPageSpecInTwips.iPortraitPageSize.iHeight, Panic(ECanonPageSpecNotSet));
	iControl = CCanonControl::NewL(this, aPrinterPort, *iStore, iModelInfo->iResourcesStreamId);
	}

CCanonControl* CCanonControl::NewL(CPdrDevice* aPdrDevice, CPrinterPort* aPrinterPort, CStreamStore& aStore, TStreamId aResourcesStreamId)
	{
	CCanonControl* control = new(ELeave) CCanonControl(aPdrDevice, aPrinterPort);
	CleanupStack::PushL(control);
	control->ConstructL(aStore, aResourcesStreamId);
	CleanupStack::Pop();
	return control;
	}

CCanonControl::~CCanonControl()
	{
	}

void CCanonControl::ConstructL(CStreamStore& aStore, TStreamId aResourcesStreamId)
	{
	if((iPdrDevice->CurrentPageSpecInTwips().iOrientation == TPageSpec::ELandscape)
		&& (iPdrDevice->Flags() & ECanonLandscapeNotAvailable))
		User::Leave(KErrNotSupported);
	CFbsDrvControl::ConstructL(aStore, aResourcesStreamId);

	TRect rect = iPdrDevice->PrintablePageInPixels();
	TSize size;
	size.iWidth = iPdrDevice->HorizontalPixelsToTwips(1000);
	size.iHeight = iPdrDevice->VerticalPixelsToTwips(1000);
	iBandedDevice = CBandedDevice::NewL(rect, size, iPdrDevice->DisplayMode(), EBandingTopToBottom, KCanonNumScanLinesPerBand);
	TInt len = CFbsBitmap::ScanLineLength(iBandedDevice->BandBitmap()->SizeInPixels().iWidth, iPdrDevice->DisplayMode());
	CFbsDrvControl::iScanLine = HBufC8::NewL(len);
	CFbsDrvControl::iCompressedScanLine = HBufC8::NewL(len);
	iPageText = CPageText::NewL();
	}

void CCanonControl::OutputBandL() 	
	{
	if(IsGraphicsBand())
		{
		TRect bandrect = iBandedDevice->BandRect();
		TSize size = bandrect.Size();
		TCommandString des;
		TBool datainband = EFalse;
		for(TInt i = 0; i < size.iWidth; i++)
			{
			iBandedDevice->BandBitmap()->GetVerticalScanLine(iScanLine, i, iPdrDevice->DisplayMode());
			if(TransformBuffer() && !datainband)
				{
				MoveToL(bandrect.iTl + TPoint(i, 0));
				des.Format(iResources->ResourceString(EPdrBitmapStart), KCanonBytesPerDotColumn * (size.iWidth - i));
				iPageBuffer->AddBytesL(des);
				datainband = ETrue;
				}
			if(datainband)
				iPageBuffer->AddBytesL(iScanLine);
			}
		iPageBuffer->AddBytesL(iResources->ResourceString(EPdrBitmapEnd));

		iPosition.iX = iPdrDevice->OffsetInPixels().iX;
		TInt numentries = iPageText->NumEntries();
		if(numentries)
			{
			CPageTextEntry* entry;
			for(TInt y = bandrect.iTl.iY; y <= bandrect.iBr.iY; y++)
				{
				for(TInt index = 0; (index < numentries); index++)
					{
					entry = (*iPageText)[index];
					TPoint drawPos = entry->iDrawPos - TPoint(0, 16);	// bodge to align pdr fonts with true types
					if(drawPos.iY == y)
						OutputTextL(drawPos, entry->iTextWidthInPixels, *(entry->iTextFormat), *(entry->iText));
					}
				}
			}
		}
	}

void CCanonControl::MoveToL(const TPoint& aPoint) 
	{
	TPoint vector = aPoint - iPosition;
	TInt maxNoOfYIncrements = 256 * 5 / 6; // Ensures no more than 256
	for(; vector.iY >= maxNoOfYIncrements; )
		{
		MoveByL(TPoint(0, maxNoOfYIncrements));
		vector = aPoint - iPosition;
		}
	MoveByL(vector);		
	iPosition = aPoint;
	}

void CCanonControl::MoveByL(const TPoint& aPoint)
	{
	TPoint vector = aPoint;
	if(vector.iX < 0)
		{
		CommandL(EPdrCarriageReturn);
		vector.iX += iPosition.iX - iPdrDevice->OffsetInPixels().iX;
		}
	vector.iX = vector.iX * 2 / 3; // X increments are in 120ths of an inch
	vector.iY = vector.iY * 6 / 5; // Y increments are in 216ths of an inch
	TPoint oldPosition = iPosition;
	CFbsDrvControl::MoveByL(vector);
	iPosition = oldPosition + aPoint; // Correct the position
	}

TBool CCanonControl::TransformBuffer()
	{
	TUint8* pStart = (TUint8*)iScanLine.Ptr();
	TUint8* pEnd = pStart + (KCanonNumScanLinesPerBand >> 3);
	TUint8* p;
	for(p = pStart; (p < pEnd) && (*p == 0xFF); p++)
		{
		}
	TBool datainscanline = (p < pEnd);
	for(p = pStart; p < pEnd; p++)
		{
		TInt byte1 = *p;
		TInt byte2 = 0;
		for(TInt j = 0; j < 8; j++)
			{
			byte2 = byte2 << 1;
			byte2 = byte2 | (byte1 & 1);
			byte1 = byte1 >> 1;
			}
		*p = (TUint8)~byte2;
		}
	return datainscanline;	// returns ETrue if there are non-blank bytes in scanline
  }

CCanonControl::CCanonControl(CPdrDevice* aPdrDevice, CPrinterPort* aPrinterPort):
	CFbsDrvControl(aPdrDevice, aPrinterPort)
	{
	__DECLARE_NAME(_S("CCanonControl"));
	}