/*
   Source File : FreeTypeFaceWrapper.h


   Copyright 2011 Gal Kahana PDFWriter

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

   
*/
#pragma once

#include "EFontStretch.h"
#include "EStatusCode.h"

#include <ft2build.h>
#include FT_FREETYPE_H

#include <utility>
#include <list>
#include <string>
#include <vector>

class IFreeTypeFaceExtender;
class IWrittenFont;
class ObjectsContext;



typedef std::pair<bool,FT_Short> BoolAndFTShort;
typedef std::list<unsigned int> UIntList;
typedef std::list<UIntList> UIntListList;
typedef std::list<std::string> StringList;
typedef std::list<unsigned long> ULongList;
typedef std::list<ULongList> ULongListList;

class FreeTypeFaceWrapper
{
public:
	class IOutlineEnumerator;

	// first overload - all but type ones. the file path is just given for storage (later queries may want it)
	FreeTypeFaceWrapper(FT_Face inFace,const std::string& inFontFilePath,long inFontIndex,bool inDoOwn = true);

	// second overload - type 1, to allow passing pfm file path. do not bother
	// if you don't have a PFM file. no i don't care about the godamn AFM file. just the PFM.
	// if you don't have a PFM i'll manage. again - i don't need the @#$@#$ AFM file. 
	// you see. i need to know if the font is serif, script 'n such. AFM - even if there
	// does not have that kind of info. so @#$@#$ off.
	// for any case, i'll check the file extension, and only do something about it if it has a pfm extension
	FreeTypeFaceWrapper(FT_Face inFace,const std::string& inFontFilePath,const std::string& inPFMFilePath,long inFontIndex,bool inDoOwn = true);
	~FreeTypeFaceWrapper(void);

	FT_Error DoneFace();

	FT_Face operator->();
	operator FT_Face();

	bool IsValid();

	PDFHummus::EStatusCode GetGlyphsForUnicodeText(const ULongList& inUnicodeCharacters,UIntList& outGlyphs);
	PDFHummus::EStatusCode GetGlyphsForUnicodeText(const ULongListList& inUnicodeCharacters,UIntListList& outGlyphs);

	std::string GetPostscriptName();
	double GetItalicAngle();
	BoolAndFTShort GetCapHeight(); // aligned to pdf metrics
	BoolAndFTShort GetxHeight(); // aligned to pdf metrics
	FT_UShort GetStemV(); // aligned to pdf metrics
	EFontStretch GetFontStretch();
	FT_UShort GetFontWeight();
	// these would be flags for the font as a whole. if subsetting, match to the character set
	unsigned int GetFontFlags();
	const char* GetTypeString();
    std::string GetGlyphName(unsigned int inGlyphIndex);
    FT_Pos GetGlyphWidth(unsigned int inGlyphIndex);
	bool GetGlyphOutline(unsigned int inGlyphIndex, IOutlineEnumerator& inEnumerator);

	// Create the written font object, matching to write this font in the best way.
	IWrittenFont* CreateWrittenFontObject(ObjectsContext* inObjectsContext);


	// flags determining values
	bool IsFixedPitch();
	bool IsSerif();
	bool IsScript();
	bool IsItalic();
	bool IsForceBold();

	// will be used externally to determine if font is symbolic or not
	bool IsCharachterCodeAdobeStandard(FT_ULong inCharacterCode);

	const std::string& GetFontFilePath();
    long GetFontIndex();


	// use this method to align measurements from (remember the dreaded point per EM!!!).
	// all measurements in this class are already aligned...so no need to align them
	FT_Short GetInPDFMeasurements(FT_Short inFontMeasurement);
	FT_UShort GetInPDFMeasurements(FT_UShort inFontMeasurement);
	FT_Pos GetInPDFMeasurements(FT_Pos inFontMeasurement);

    // translate to freetype glyph index. this may not always be the input, as in the case
    // of type 3 fonts with private encoding. so what actually gets in is the glyph index
    // according to hummus
    unsigned int GetGlyphIndexInFreeTypeIndexes(unsigned int inGlyphIndex);

	// load glyph, use the freetype load glyph, but add some extra flags that are common to 
	// Hummus requirements, which is SPEED SPEED. as i don't normally need rendering
	// hinting and scaling is always removed. inFlags are added on top of the default flags
	FT_Error LoadGlyph(FT_UInt inGlyphIndex, FT_Int32 inFlags = 0);

private:

	FT_Face mFace;
	IFreeTypeFaceExtender* mFormatParticularWrapper;
	bool mHaslowercase;
	std::string mFontFilePath;
    long mFontIndex;
    std::string mNotDefGlyphName;
	bool mGlyphIsLoaded;
	unsigned int mCurrentGlyph;
	bool mDoesOwn;

	BoolAndFTShort GetCapHeightInternal(); 
	BoolAndFTShort GetxHeightInternal(); 
	FT_UShort GetStemVInternal(); 


	std::string GetExtension(const std::string& inFilePath);
	void SetupFormatSpecificExtender(const std::string& inFilePath, const std::string& inPFMFilePath);
	BoolAndFTShort CapHeightFromHHeight();
	BoolAndFTShort XHeightFromLowerXHeight();
	BoolAndFTShort GetYBearingForUnicodeChar(unsigned short unicodeCharCode);
	EFontStretch StretchFromName();
	FT_UShort WeightFromName();
	bool IsSymbolic();
	bool IsDefiningCharsNotInAdobeStandardLatin();
	std::string NotDefGlyphName();

public:
	class IOutlineEnumerator {
	public:
		IOutlineEnumerator(){ mUPM = 0; }
		virtual ~IOutlineEnumerator(){};

		inline FT_Short UPM() const { return mUPM; } //typically 1000 or 2048

	protected:
		//in font em units relative to UPM
		virtual bool Moveto(FT_Short x, FT_Short y) =0; //ret false to abort
		virtual bool Lineto(FT_Short x, FT_Short y) =0;
		virtual bool Curveto(FT_Short x1, FT_Short y1, FT_Short x2, FT_Short y2, FT_Short x3, FT_Short y3) =0;
		virtual bool Close() =0;

	private:
		friend bool FreeTypeFaceWrapper::GetGlyphOutline(unsigned int inGlyphIndex, IOutlineEnumerator& inEnumerator);
		static int outline_moveto(const FT_Vector* to, void *closure);
		static int outline_lineto(const FT_Vector* to, void *closure);
		static int outline_conicto(const FT_Vector *control, const FT_Vector *to, void *closure);
		static int outline_cubicto(const FT_Vector *control1, const FT_Vector *control2, const FT_Vector *to, void *closure);

		void FTBegin(FT_UShort upm);
		inline bool FTMoveto(const FT_Vector *to); //rets true if ok
		inline bool FTLineto(const FT_Vector *to);
		inline bool FTConicto(const FT_Vector *control, const FT_Vector *to);
		inline bool FTCubicto(const FT_Vector *control1, const FT_Vector *control2, const FT_Vector *to);
		void FTEnd();

		FT_UShort mUPM; //units per EM
		bool mToLastValid;
		FT_Vector mToLast;
	};
};

