1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-04 18:06:22 +03:00

0029122: Visualization - improve Font_BRepFont to handle one-line-fonts

Font_SystemFont - added a new property SingleStrokeFont().
Font_BRepFont::renderGlyph() has been extended to not close contours
when flag SingleStrokeFont() has been set.
This commit is contained in:
kgv 2017-09-19 16:53:39 +03:00 committed by bugmaster
parent 1a6726dabb
commit e4f0cc46a0
8 changed files with 254 additions and 199 deletions

View File

@ -265,7 +265,7 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
TopLoc_Location aLoc; TopLoc_Location aLoc;
TopoDS_Face aFaceDraft; TopoDS_Face aFaceDraft;
myBuilder.MakeFace (aFaceDraft, mySurface, myPrecision); TopoDS_Compound aFaceCompDraft;
// Get orientation is useless since it doesn't retrieve any in-font information and just computes orientation. // Get orientation is useless since it doesn't retrieve any in-font information and just computes orientation.
// Because it fails in some cases - leave this to ShapeFix. // Because it fails in some cases - leave this to ShapeFix.
@ -277,7 +277,7 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
const short anEndIndex = anOutline.contours[aContour]; const short anEndIndex = anOutline.contours[aContour];
const short aPntsNb = (anEndIndex - aStartIndex) + 1; const short aPntsNb = (anEndIndex - aStartIndex) + 1;
aStartIndex = anEndIndex + 1; aStartIndex = anEndIndex + 1;
if (aPntsNb < 3) if (aPntsNb < 3 && !myIsSingleLine)
{ {
// closed contour can not be constructed from < 3 points // closed contour can not be constructed from < 3 points
continue; continue;
@ -289,7 +289,8 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
gp_XY aPntCurr = readFTVec (aPntList[aPntsNb - 1], myScaleUnits); gp_XY aPntCurr = readFTVec (aPntList[aPntsNb - 1], myScaleUnits);
gp_XY aPntNext = readFTVec (aPntList[0], myScaleUnits); gp_XY aPntNext = readFTVec (aPntList[0], myScaleUnits);
Standard_Integer aLinePnts = (FT_CURVE_TAG(aTags[aPntsNb - 1]) == FT_Curve_Tag_On) ? 1 : 0; bool isLineSeg = !myIsSingleLine
&& FT_CURVE_TAG(aTags[aPntsNb - 1]) == FT_Curve_Tag_On;
gp_XY aPntLine1 = aPntCurr; gp_XY aPntLine1 = aPntCurr;
// see http://freetype.sourceforge.net/freetype2/docs/glyphs/glyphs-6.html // see http://freetype.sourceforge.net/freetype2/docs/glyphs/glyphs-6.html
@ -303,10 +304,10 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
// process tags // process tags
if (FT_CURVE_TAG(aTags[aPntId]) == FT_Curve_Tag_On) if (FT_CURVE_TAG(aTags[aPntId]) == FT_Curve_Tag_On)
{ {
if (aLinePnts < 1) if (!isLineSeg)
{ {
aPntLine1 = aPntCurr; aPntLine1 = aPntCurr;
aLinePnts = 1; isLineSeg = true;
continue; continue;
} }
@ -315,7 +316,7 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
if (aLen <= myPrecision) if (aLen <= myPrecision)
{ {
aPntLine1 = aPntCurr; aPntLine1 = aPntCurr;
aLinePnts = 1; isLineSeg = true;
continue; continue;
} }
@ -339,7 +340,7 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
} }
else if (FT_CURVE_TAG(aTags[aPntId]) == FT_Curve_Tag_Conic) else if (FT_CURVE_TAG(aTags[aPntId]) == FT_Curve_Tag_Conic)
{ {
aLinePnts = 0; isLineSeg = false;
gp_XY aPntPrev2 = aPntPrev; gp_XY aPntPrev2 = aPntPrev;
gp_XY aPntNext2 = aPntNext; gp_XY aPntNext2 = aPntNext;
@ -378,7 +379,7 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
else if (FT_CURVE_TAG(aTags[aPntId]) == FT_Curve_Tag_Cubic else if (FT_CURVE_TAG(aTags[aPntId]) == FT_Curve_Tag_Cubic
&& FT_CURVE_TAG(aTags[(aPntId + 1) % aPntsNb]) == FT_Curve_Tag_Cubic) && FT_CURVE_TAG(aTags[(aPntId + 1) % aPntsNb]) == FT_Curve_Tag_Cubic)
{ {
aLinePnts = 0; isLineSeg = false;
my4Poles.SetValue (1, aPntPrev); my4Poles.SetValue (1, aPntPrev);
my4Poles.SetValue (2, aPntCurr); my4Poles.SetValue (2, aPntCurr);
my4Poles.SetValue (3, aPntNext); my4Poles.SetValue (3, aPntNext);
@ -411,7 +412,8 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
const gp_Pnt2d aFirstPnt = aDraft2d->StartPoint(); const gp_Pnt2d aFirstPnt = aDraft2d->StartPoint();
const gp_Pnt2d aLastPnt = aDraft2d->EndPoint(); const gp_Pnt2d aLastPnt = aDraft2d->EndPoint();
if (!aFirstPnt.IsEqual (aLastPnt, myPrecision)) if (!myIsSingleLine
&& !aFirstPnt.IsEqual (aLastPnt, myPrecision))
{ {
Handle(Geom2d_TrimmedCurve) aLine = GCE2d_MakeSegment (aLastPnt, aFirstPnt); Handle(Geom2d_TrimmedCurve) aLine = GCE2d_MakeSegment (aLastPnt, aFirstPnt);
myConcatMaker.Add (aLine, myPrecision); myConcatMaker.Add (aLine, myPrecision);
@ -438,7 +440,8 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
TopExp::Vertices (aWireMaker.Wire(), aFirstV, aLastV); TopExp::Vertices (aWireMaker.Wire(), aFirstV, aLastV);
gp_Pnt aFirstPoint = BRep_Tool::Pnt (aFirstV); gp_Pnt aFirstPoint = BRep_Tool::Pnt (aFirstV);
gp_Pnt aLastPoint = BRep_Tool::Pnt (aLastV); gp_Pnt aLastPoint = BRep_Tool::Pnt (aLastV);
if (!aFirstPoint.IsEqual (aLastPoint, myPrecision)) if (!myIsSingleLine
&& !aFirstPoint.IsEqual (aLastPoint, myPrecision))
{ {
aWireMaker.Add (BRepLib_MakeEdge (aFirstV, aLastV)); aWireMaker.Add (BRepLib_MakeEdge (aFirstV, aLastV));
} }
@ -450,31 +453,64 @@ Standard_Boolean Font_BRepFont::renderGlyph (const Standard_Utf32Char theChar,
} }
TopoDS_Wire aWireDraft = aWireMaker.Wire(); TopoDS_Wire aWireDraft = aWireMaker.Wire();
//if (anOrient == FT_ORIENTATION_FILL_LEFT) if (!myIsSingleLine)
//{ {
// According to the TrueType specification, clockwise contours must be filled //if (anOrient == FT_ORIENTATION_FILL_LEFT)
aWireDraft.Reverse(); //{
//} // According to the TrueType specification, clockwise contours must be filled
myBuilder.Add (aFaceDraft, aWireDraft); aWireDraft.Reverse();
//}
if (aFaceDraft.IsNull())
{
myBuilder.MakeFace (aFaceDraft, mySurface, myPrecision);
}
myBuilder.Add (aFaceDraft, aWireDraft);
}
else
{
if (aFaceCompDraft.IsNull())
{
myBuilder.MakeCompound (aFaceCompDraft);
}
myBuilder.Add (aFaceCompDraft, aWireDraft);
}
} }
myFixer.Init (aFaceDraft); if (!aFaceDraft.IsNull())
myFixer.Perform();
theShape = myFixer.Result();
if (!theShape.IsNull()
&& theShape.ShapeType() != TopAbs_FACE)
{ {
// shape fix can not fix orientation within the single call myFixer.Init (aFaceDraft);
TopoDS_Compound aComp; myFixer.Perform();
myBuilder.MakeCompound (aComp); TopoDS_Shape aFixResult = myFixer.Result();
for (TopExp_Explorer aFaceIter (theShape, TopAbs_FACE); aFaceIter.More(); aFaceIter.Next()) if (!aFixResult.IsNull()
&& aFixResult.ShapeType() != TopAbs_FACE)
{ {
TopoDS_Face aFace = TopoDS::Face (aFaceIter.Current()); // shape fix can not fix orientation within the single call
myFixer.Init (aFace); if (aFaceCompDraft.IsNull())
myFixer.Perform(); {
myBuilder.Add (aComp, myFixer.Result()); myBuilder.MakeCompound (aFaceCompDraft);
}
for (TopExp_Explorer aFaceIter (aFixResult, TopAbs_FACE); aFaceIter.More(); aFaceIter.Next())
{
TopoDS_Face aFace = TopoDS::Face (aFaceIter.Current());
myFixer.Init (aFace);
myFixer.Perform();
myBuilder.Add (aFaceCompDraft, myFixer.Result());
}
theShape = aFaceCompDraft;
} }
theShape = aComp; else if (!aFaceCompDraft.IsNull())
{
myBuilder.Add (aFaceCompDraft, aFixResult);
theShape = aFaceCompDraft;
}
else
{
theShape = aFixResult;
}
}
else if (!aFaceCompDraft.IsNull())
{
theShape = aFaceCompDraft;
} }
myCache.Bind (theChar, theShape); myCache.Bind (theChar, theShape);

View File

@ -33,6 +33,7 @@ Font_FTFont::Font_FTFont (const Handle(Font_FTLibrary)& theFTLib)
myFTFace (NULL), myFTFace (NULL),
myPointSize (0U), myPointSize (0U),
myLoadFlags (FT_LOAD_NO_HINTING | FT_LOAD_TARGET_NORMAL), myLoadFlags (FT_LOAD_NO_HINTING | FT_LOAD_TARGET_NORMAL),
myIsSingleLine(false),
myKernAdvance(new FT_Vector()), myKernAdvance(new FT_Vector()),
myUChar (0U) myUChar (0U)
{ {
@ -118,9 +119,12 @@ bool Font_FTFont::Init (const NCollection_String& theFontName,
{ {
Handle(Font_FontMgr) aFontMgr = Font_FontMgr::GetInstance(); Handle(Font_FontMgr) aFontMgr = Font_FontMgr::GetInstance();
const Handle(TCollection_HAsciiString) aFontName = new TCollection_HAsciiString (theFontName.ToCString()); const Handle(TCollection_HAsciiString) aFontName = new TCollection_HAsciiString (theFontName.ToCString());
Handle(Font_SystemFont) aRequestedFont = aFontMgr->FindFont (aFontName, theFontAspect, thePointSize); if (Handle(Font_SystemFont) aRequestedFont = aFontMgr->FindFont (aFontName, theFontAspect, thePointSize))
return !aRequestedFont.IsNull() {
&& Font_FTFont::Init (aRequestedFont->FontPath()->ToCString(), thePointSize, theResolution); myIsSingleLine = aRequestedFont->IsSingleStrokeFont();
return Font_FTFont::Init (aRequestedFont->FontPath()->ToCString(), thePointSize, theResolution);
}
return false;
} }
// ======================================================================= // =======================================================================

View File

@ -73,6 +73,13 @@ public:
const unsigned int thePointSize, const unsigned int thePointSize,
const unsigned int theResolution); const unsigned int theResolution);
//! Return TRUE if this is single-stroke (one-line) font, FALSE by default.
//! Such fonts define single-line glyphs instead of closed contours, so that they are rendered incorrectly by normal software.
bool IsSingleStrokeFont() const { return myIsSingleLine; }
//! Set if this font should be rendered as single-stroke (one-line).
void SetSingleStrokeFont (bool theIsSingleLine) { myIsSingleLine = theIsSingleLine; }
//! Release currently loaded font. //! Release currently loaded font.
Standard_EXPORT virtual void Release(); Standard_EXPORT virtual void Release();
@ -159,6 +166,7 @@ protected:
NCollection_String myFontPath; //!< font path NCollection_String myFontPath; //!< font path
unsigned int myPointSize; //!< point size set by FT_Set_Char_Size unsigned int myPointSize; //!< point size set by FT_Set_Char_Size
int32_t myLoadFlags; //!< default load flags int32_t myLoadFlags; //!< default load flags
bool myIsSingleLine;//!< single stroke font flag, FALSE by default
Image_PixMap myGlyphImg; //!< cached glyph plane Image_PixMap myGlyphImg; //!< cached glyph plane
FT_Vector* myKernAdvance; //!< buffer variable FT_Vector* myKernAdvance; //!< buffer variable

View File

@ -218,6 +218,8 @@ static Handle(Font_SystemFont) checkFont (const Handle(Font_FTLibrary)& theFTLib
Handle(TCollection_HAsciiString) aFontName = new TCollection_HAsciiString (aFontFace->family_name); Handle(TCollection_HAsciiString) aFontName = new TCollection_HAsciiString (aFontFace->family_name);
Handle(TCollection_HAsciiString) aFontPath = new TCollection_HAsciiString (theFontPath); Handle(TCollection_HAsciiString) aFontPath = new TCollection_HAsciiString (theFontPath);
aResult = new Font_SystemFont (aFontName, anAspect, aFontPath); aResult = new Font_SystemFont (aFontName, anAspect, aFontPath);
// automatically identify some known single-line fonts
aResult->SetSingleStrokeFont (aFontName->String().StartsWith ("OLF "));
} }
FT_Done_Face (aFontFace); FT_Done_Face (aFontFace);

View File

@ -13,126 +13,113 @@
// Alternatively, this file may be used under the terms of Open CASCADE // Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement. // commercial license or contractual agreement.
// Updated:
#include <Font_SystemFont.hxx> #include <Font_SystemFont.hxx>
#include <OSD_Path.hxx> #include <OSD_Path.hxx>
#include <Standard_Type.hxx> #include <Standard_Type.hxx>
#include <TCollection_HAsciiString.hxx> #include <TCollection_HAsciiString.hxx>
IMPLEMENT_STANDARD_RTTIEXT(Font_SystemFont,Standard_Transient) IMPLEMENT_STANDARD_RTTIEXT(Font_SystemFont, Standard_Transient)
Font_SystemFont::Font_SystemFont(): // =======================================================================
MyFontName(), // function : Font_SystemFont
MyFontAspect(Font_FA_Undefined), // purpose :
MyFaceSize(-1), // =======================================================================
MyVerification(Standard_False) Font_SystemFont::Font_SystemFont()
: myFontAspect (Font_FA_Undefined),
myFaceSize (-1),
myIsSingleLine (Standard_False),
myIsDefined (Standard_False)
{ {
//
} }
Font_SystemFont::Font_SystemFont( const Handle(TCollection_HAsciiString)& FontName, // =======================================================================
const Font_FontAspect FontAspect, // function : Font_SystemFont
const Handle(TCollection_HAsciiString)& FilePath ): // purpose :
MyFontName(FontName), // =======================================================================
MyFontAspect(FontAspect), Font_SystemFont::Font_SystemFont (const Handle(TCollection_HAsciiString)& theFontName,
MyFaceSize(-1), const Font_FontAspect theFontAspect,
MyFilePath(FilePath), const Handle(TCollection_HAsciiString)& theFilePath)
MyVerification(Standard_True) : myFontName (theFontName),
myFontAspect (theFontAspect),
myFaceSize (-1),
myFilePath (theFilePath),
myIsSingleLine (Standard_False),
myIsDefined (Standard_True)
{ {
//
} }
// =======================================================================
// function : Font_SystemFont
// purpose :
// =======================================================================
Font_SystemFont::Font_SystemFont (const Handle(TCollection_HAsciiString)& theXLFD, Font_SystemFont::Font_SystemFont (const Handle(TCollection_HAsciiString)& theXLFD,
const Handle(TCollection_HAsciiString)& theFilePath) : const Handle(TCollection_HAsciiString)& theFilePath)
MyFontAspect(Font_FA_Regular), : myFontAspect (Font_FA_Regular),
MyFaceSize(-1), myFaceSize (-1),
MyFilePath(theFilePath) myFilePath (theFilePath),
myIsSingleLine (Standard_False),
myIsDefined (Standard_True)
{ {
MyVerification = Standard_True; if (theXLFD.IsNull()
if (theXLFD.IsNull()) || theXLFD->IsEmpty())
{ {
MyVerification = Standard_False; // empty font description handler myIsDefined = Standard_False;
} return;
if (theXLFD->IsEmpty())
{
MyVerification = Standard_False; // empty font description
} }
if (MyVerification) myFontName = theXLFD->Token ("-", 2);
const TCollection_AsciiString& aXLFD = theXLFD->String();
// Getting font size for fixed size fonts
if (aXLFD.Search ("-0-0-0-0-") >= 0)
{ {
MyFontName = theXLFD->Token ("-", 2); myFaceSize = -1; // Scalable font
TCollection_AsciiString aXLFD (theXLFD->ToCString()); }
else
{
myFaceSize = aXLFD.Token ("-", 7).IntegerValue();
}
// Getting font size for fixed size fonts // Detect font aspect
if (aXLFD.Search ("-0-0-0-0-") >= 0) if (aXLFD.Token ("-", 3).IsEqual ("bold")
MyFaceSize = -1; // Scalable font && (aXLFD.Token ("-", 4).IsEqual ("i")
else || aXLFD.Token ("-", 4).IsEqual ("o")))
//TODO catch exeption {
MyFaceSize = aXLFD.Token ("-", 7).IntegerValue(); myFontAspect = Font_FA_BoldItalic;
}
// Detect font aspect else if (aXLFD.Token ("-", 3).IsEqual ("bold"))
if (aXLFD.Token ("-", 3).IsEqual ("bold") && {
(aXLFD.Token ("-", 4).IsEqual ("i") || aXLFD.Token ("-", 4).IsEqual ("o"))) myFontAspect = Font_FA_Bold;
{ }
MyFontAspect = Font_FA_BoldItalic; else if (aXLFD.Token ("-", 4).IsEqual ("i")
} || aXLFD.Token ("-", 4).IsEqual ("o"))
else if (aXLFD.Token ("-", 3).IsEqual ("bold")) {
{ myFontAspect = Font_FA_Italic;
MyFontAspect = Font_FA_Bold;
}
else if (aXLFD.Token ("-", 4).IsEqual ("i") || aXLFD.Token ("-", 4).IsEqual ("o"))
{
MyFontAspect = Font_FA_Italic;
}
} }
} }
Standard_Boolean Font_SystemFont::IsValid() const{ // =======================================================================
if ( !MyVerification) // function : IsValid
return Standard_False; // purpose :
// =======================================================================
if ( MyFontAspect == Font_FA_Undefined ) Standard_Boolean Font_SystemFont::IsValid() const
return Standard_False;
if ( MyFontName->IsEmpty() || !MyFontName->IsAscii() )
return Standard_False;
OSD_Path path;
return path.IsValid( MyFilePath->String() );
}
Handle(TCollection_HAsciiString) Font_SystemFont::FontPath() const{
return MyFilePath;
}
Handle(TCollection_HAsciiString) Font_SystemFont::FontName() const{
return MyFontName;
}
Font_FontAspect Font_SystemFont::FontAspect() const{
return MyFontAspect;
}
Standard_Integer Font_SystemFont::FontHeight() const {
return MyFaceSize;
}
Standard_Boolean Font_SystemFont::IsEqual(const Handle(Font_SystemFont)& theOtherFont) const
{ {
if (!MyFontName->IsSameString (theOtherFont->FontName(), Standard_False)) return myIsDefined
{ && myFontAspect != Font_FA_Undefined
return Standard_False; && !myFontName->IsEmpty()
} && OSD_Path::IsValid (myFilePath->String());
}
if (MyFontAspect != theOtherFont->FontAspect())
{ // =======================================================================
return Standard_False; // function : IsEqual
} // purpose :
// =======================================================================
if (MyFaceSize != theOtherFont->FontHeight()) Standard_Boolean Font_SystemFont::IsEqual (const Handle(Font_SystemFont)& theOtherFont) const
{ {
return Standard_False; return myFontName->IsSameString (myFontName, Standard_False)
} && myFontAspect == theOtherFont->myFontAspect
&& myFaceSize == theOtherFont->myFaceSize;
return Standard_True;
} }

View File

@ -25,76 +25,60 @@
#include <Standard_Transient.hxx> #include <Standard_Transient.hxx>
class TCollection_HAsciiString; class TCollection_HAsciiString;
//! This class stores information about the font, which is merely a file path and cached metadata about the font.
class Font_SystemFont;
DEFINE_STANDARD_HANDLE(Font_SystemFont, Standard_Transient)
//! Structure for store of Font System Information
class Font_SystemFont : public Standard_Transient class Font_SystemFont : public Standard_Transient
{ {
DEFINE_STANDARD_RTTIEXT(Font_SystemFont, Standard_Transient)
public: public:
//! Creates an empty font object.
//! Creates empty font object
Standard_EXPORT Font_SystemFont(); Standard_EXPORT Font_SystemFont();
//! Creates Font object initialized with <FontName> as name //! Creates a new font object.
//! <FontAspect>.... TODO Standard_EXPORT Font_SystemFont (const Handle(TCollection_HAsciiString)& theFontName,
Standard_EXPORT Font_SystemFont(const Handle(TCollection_HAsciiString)& theFontName, const Font_FontAspect theFontAspect, const Handle(TCollection_HAsciiString)& theFilePath); const Font_FontAspect theFontAspect,
const Handle(TCollection_HAsciiString)& theFilePath);
//! Creates Font object and initialize class fields with //! Creates a font object and initialize class fields with values taken from XLFD (X Logical Font Description)
//! values taken from XLFD (X Logical Font Description) Standard_EXPORT Font_SystemFont (const Handle(TCollection_HAsciiString)& theXLFD,
Standard_EXPORT Font_SystemFont(const Handle(TCollection_HAsciiString)& theXLFD, const Handle(TCollection_HAsciiString)& theFilePath); const Handle(TCollection_HAsciiString)& theFilePath);
//! Returns font family name //! Returns font family name.
Standard_EXPORT Handle(TCollection_HAsciiString) FontName() const; const Handle(TCollection_HAsciiString)& FontName() const { return myFontName; }
//! Returns font file path //! Returns font file path.
//! Level: Public const Handle(TCollection_HAsciiString)& FontPath() const { return myFilePath; }
Standard_EXPORT Handle(TCollection_HAsciiString) FontPath() const;
//! Returns font aspect //! Returns font aspect.
//! Level: Public Font_FontAspect FontAspect() const { return myFontAspect; }
Standard_EXPORT Font_FontAspect FontAspect() const;
//! Returns font height //! Returns font height.
//! If returned value is equal -1 it means that font is resizable //! If returned value is equal -1 it means that font is resizable.
//! Level: Public Standard_Integer FontHeight() const { return myFaceSize; }
Standard_EXPORT Standard_Integer FontHeight() const;
Standard_EXPORT Standard_Boolean IsValid() const; Standard_EXPORT Standard_Boolean IsValid() const;
//! Return true if the FontName, FontAspect and FontSize are the same. //! Return true if the FontName, FontAspect and FontSize are the same.
//! Level: Public
Standard_EXPORT Standard_Boolean IsEqual (const Handle(Font_SystemFont)& theOtherFont) const; Standard_EXPORT Standard_Boolean IsEqual (const Handle(Font_SystemFont)& theOtherFont) const;
//! Return TRUE if this is single-stroke (one-line) font, FALSE by default.
//! Such fonts define single-line glyphs instead of closed contours, so that they are rendered incorrectly by normal software.
Standard_Boolean IsSingleStrokeFont() const { return myIsSingleLine; }
//! Set if this font should be rendered as single-stroke (one-line).
void SetSingleStrokeFont (Standard_Boolean theIsSingleLine) { myIsSingleLine = theIsSingleLine; }
DEFINE_STANDARD_RTTIEXT(Font_SystemFont,Standard_Transient)
protected:
private: private:
Handle(TCollection_HAsciiString) myFontName;
Handle(TCollection_HAsciiString) MyFontName; Font_FontAspect myFontAspect;
Font_FontAspect MyFontAspect; Standard_Integer myFaceSize;
Standard_Integer MyFaceSize; Handle(TCollection_HAsciiString) myFilePath;
Handle(TCollection_HAsciiString) MyFilePath; Standard_Boolean myIsSingleLine; //!< single stroke font flag, FALSE by default
Standard_Boolean MyVerification; Standard_Boolean myIsDefined;
}; };
DEFINE_STANDARD_HANDLE(Font_SystemFont, Standard_Transient)
#endif // _Font_SystemFont_HeaderFile #endif // _Font_SystemFont_HeaderFile

View File

@ -5594,26 +5594,37 @@ static int VFont (Draw_Interpretor& theDI,
{ {
if (++anArgIter >= theArgNb) if (++anArgIter >= theArgNb)
{ {
std::cerr << "Wrong syntax at argument '" << anArg.ToCString() << "'!\n"; std::cerr << "Error: wrong syntax at argument '" << anArg << "'!\n";
return 1; return 1;
} }
Standard_CString aFontPath = theArgVec[anArgIter];
Standard_CString aFontName = NULL; Standard_CString aFontPath = theArgVec[anArgIter++];
TCollection_AsciiString aFontName;
Font_FontAspect aFontAspect = Font_FA_Undefined; Font_FontAspect aFontAspect = Font_FA_Undefined;
if (++anArgIter < theArgNb) Standard_Integer isSingelStroke = -1;
for (; anArgIter < theArgNb; ++anArgIter)
{ {
if (!parseFontStyle (anArgCase, aFontAspect)) anArgCase = theArgVec[anArgIter];
anArgCase.LowerCase();
if (aFontAspect == Font_FA_Undefined
&& parseFontStyle (anArgCase, aFontAspect))
{
continue;
}
else if (anArgCase == "singlestroke"
|| anArgCase == "singleline"
|| anArgCase == "oneline")
{
isSingelStroke = 1;
}
else if (aFontName.IsEmpty())
{ {
aFontName = theArgVec[anArgIter]; aFontName = theArgVec[anArgIter];
} }
if (++anArgIter < theArgNb) else
{ {
anArgCase = theArgVec[anArgIter]; --anArgIter;
anArgCase.LowerCase(); break;
if (!parseFontStyle (anArgCase, aFontAspect))
{
--anArgIter;
}
} }
} }
@ -5625,19 +5636,23 @@ static int VFont (Draw_Interpretor& theDI,
} }
if (aFontAspect != Font_FA_Undefined if (aFontAspect != Font_FA_Undefined
|| aFontName != NULL) || !aFontName.IsEmpty())
{ {
if (aFontAspect == Font_FA_Undefined) if (aFontAspect == Font_FA_Undefined)
{ {
aFontAspect = aFont->FontAspect(); aFontAspect = aFont->FontAspect();
} }
Handle(TCollection_HAsciiString) aName = aFont->FontName(); Handle(TCollection_HAsciiString) aName = aFont->FontName();
if (aFontName != NULL) if (!aFontName.IsEmpty())
{ {
aName = new TCollection_HAsciiString (aFontName); aName = new TCollection_HAsciiString (aFontName);
} }
aFont = new Font_SystemFont (aName, aFontAspect, new TCollection_HAsciiString (aFontPath)); aFont = new Font_SystemFont (aName, aFontAspect, new TCollection_HAsciiString (aFontPath));
} }
if (isSingelStroke != -1)
{
aFont->SetSingleStrokeFont (isSingelStroke == 1);
}
aMgr->RegisterFont (aFont, Standard_True); aMgr->RegisterFont (aFont, Standard_True);
theDI << aFont->FontName()->String() theDI << aFont->FontName()->String()
@ -5646,7 +5661,7 @@ static int VFont (Draw_Interpretor& theDI,
} }
else else
{ {
std::cerr << "Warning! Unknown argument '" << anArg.ToCString() << "'\n"; std::cerr << "Warning! Unknown argument '" << anArg << "'\n";
} }
} }
@ -6721,7 +6736,7 @@ void ViewerTest::ObjectCommands(Draw_Interpretor& theCommands)
"\n\t\t: [-plane NormX NormY NormZ DirX DirY DirZ]", "\n\t\t: [-plane NormX NormY NormZ DirX DirY DirZ]",
__FILE__, TextToBRep, group); __FILE__, TextToBRep, group);
theCommands.Add ("vfont", theCommands.Add ("vfont",
"vfont [add pathToFont [fontName] [regular,bold,italic,bolditalic=undefined]]" "vfont [add pathToFont [fontName] [regular,bold,italic,bolditalic=undefined] [singleStroke]]"
"\n\t\t: [find fontName [regular,bold,italic,bolditalic=undefined]]", "\n\t\t: [find fontName [regular,bold,italic,bolditalic=undefined]]",
__FILE__, VFont, group); __FILE__, VFont, group);

19
tests/3rdparty/fonts/B6 vendored Normal file
View File

@ -0,0 +1,19 @@
puts "============"
puts "0029122: Visualization - improve Font_BRepFont to handle one-line-fonts"
puts "============"
puts ""
pload MODELING VISUALIZATION
vfont add [locate_data_file OLFTestFont-Regular.ttf]
vfont add [locate_data_file machtgth.ttf] singleStroke machtgth
vfont add [locate_data_file DejaVuSans.ttf] SansFont
text2brep s1 "ABCDabcd123" -font "OLF TestFont" -height 48 -pos 0 0 0
text2brep s2 "ABCDabcd123" -font "machtgth" -height 48 -pos 0 50 0
text2brep s3 "ABCDabcd123" -font "SansFont" -height 48 -pos 0 100 0
vclear
vinit View1
vtop
vdisplay -dispMode 1 s1 s2 s3
vfit
vdump ${imagedir}/${casename}.png