X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/81c67e27fd6a168e26e5acba949eca36cd746421..300b1288331acc7ac3482595bb226a52f8aec238:/src/generic/dcpsg.cpp diff --git a/src/generic/dcpsg.cpp b/src/generic/dcpsg.cpp index bc1b78711d..cdba095cf4 100644 --- a/src/generic/dcpsg.cpp +++ b/src/generic/dcpsg.cpp @@ -1142,6 +1142,10 @@ void wxPostScriptDC::SetBrush( const wxBrush& brush ) void wxPostScriptDC::DoDrawText( const wxString& text, wxCoord x, wxCoord y ) { wxCHECK_RET( m_ok && m_pstream, wxT("invalid postscript dc") ); + + wxCoord text_w, text_h, text_descent; + + GetTextExtent(text, &text_w, &text_h, &text_descent); SetFont( m_font ); @@ -1183,7 +1187,10 @@ void wxPostScriptDC::DoDrawText( const wxString& text, wxCoord x, wxCoord y ) int size = m_font.GetPointSize(); - wxCoord by = y + (wxCoord)floor( double(size) * 2.0 / 3.0 ); // approximate baseline +// wxCoord by = y + (wxCoord)floor( double(size) * 2.0 / 3.0 ); // approximate baseline +// commented by V. Slavik and replaced by accurate version +// - note that there is still rounding error in text_descent! + wxCoord by = y + size - text_descent; // baseline fprintf( m_pstream, "%d %d moveto\n", XLOG2DEV(x), YLOG2DEV(by) ); /* I don't know how to write char to a stream, so I use a mini string */ @@ -1221,8 +1228,6 @@ void wxPostScriptDC::DoDrawText( const wxString& text, wxCoord x, wxCoord y ) if (m_font.GetUnderlined()) { wxCoord uy = (wxCoord)(y + size - m_underlinePosition); - wxCoord w, h; - GetTextExtent(text, &w, &h); fprintf( m_pstream, "gsave\n" @@ -1233,13 +1238,123 @@ void wxPostScriptDC::DoDrawText( const wxString& text, wxCoord x, wxCoord y ) "grestore\n", XLOG2DEV(x), YLOG2DEV(uy), (wxCoord)m_underlineThickness, - XLOG2DEV(x + w), YLOG2DEV(uy) ); + XLOG2DEV(x + text_w), YLOG2DEV(uy) ); } CalcBoundingBox( x, y ); CalcBoundingBox( x + size * text.Length() * 2/3 , y ); } +void wxPostScriptDC::DoDrawRotatedText( const wxString& text, wxCoord x, wxCoord y, double angle ) +{ + if (angle == 0.0) + { + DoDrawText(text, x, y); + return; + } + + wxCHECK_RET( m_ok && m_pstream, wxT("invalid postscript dc") ); + + SetFont( m_font ); + + if (m_textForegroundColour.Ok()) + { + unsigned char red = m_textForegroundColour.Red(); + unsigned char blue = m_textForegroundColour.Blue(); + unsigned char green = m_textForegroundColour.Green(); + + if (!m_colour) + { + // Anything not white is black + if (! (red == (unsigned char) 255 && + blue == (unsigned char) 255 && + green == (unsigned char) 255)) + { + red = (unsigned char) 0; + green = (unsigned char) 0; + blue = (unsigned char) 0; + } + } + + // maybe setgray here ? + if (!(red == m_currentRed && green == m_currentGreen && blue == m_currentBlue)) + { + double redPS = (double)(red) / 255.0; + double bluePS = (double)(blue) / 255.0; + double greenPS = (double)(green) / 255.0; + + fprintf( m_pstream, + "%.8f %.8f %.8f setrgbcolor\n", + redPS, greenPS, bluePS ); + + m_currentRed = red; + m_currentBlue = blue; + m_currentGreen = green; + } + } + + int size = m_font.GetPointSize(); + + long by = y + (long)floor( double(size) * 2.0 / 3.0 ); // approximate baseline + + // FIXME only correct for 90 degrees + fprintf( m_pstream, "%d %d moveto\n", XLOG2DEV(x + size), YLOG2DEV(by) ); + fprintf(m_pstream, "%.8f rotate\n", angle); + + /* I don't know how to write char to a stream, so I use a mini string */ + char tmpbuf[2]; + tmpbuf[1] = 0; + + fprintf( m_pstream, "(" ); + const wxWX2MBbuf textbuf = text.mb_str(); + int len = strlen(textbuf); + int i; + for (i = 0; i < len; i++) + { + int c = (unsigned char) textbuf[i]; + if (c == ')' || c == '(' || c == '\\') + { + /* Cope with special characters */ + fprintf( m_pstream, "\\" ); + tmpbuf[0] = (char) c; + fprintf( m_pstream, tmpbuf ); + } + else if ( c >= 128 ) + { + /* Cope with character codes > 127 */ + fprintf(m_pstream, "\\%o", c); + } + else + { + tmpbuf[0] = (char) c; + fprintf( m_pstream, tmpbuf ); + } + } + + fprintf( m_pstream, ") show\n" ); + fprintf( m_pstream, "%.8f rotate\n", -angle ); + + if (m_font.GetUnderlined()) + { + long uy = (long)(y + size - m_underlinePosition); + long w, h; + GetTextExtent(text, &w, &h); + + fprintf( m_pstream, + "gsave\n" + "%d %d moveto\n" + "%ld setlinewidth\n" + "%d %d lineto\n" + "stroke\n" + "grestore\n", + XLOG2DEV(x), YLOG2DEV(uy), + (long)m_underlineThickness, + XLOG2DEV(x + w), YLOG2DEV(uy) ); + } + + CalcBoundingBox( x, y ); + CalcBoundingBox( x + size * text.Length() * 2/3 , y ); +} void wxPostScriptDC::SetBackground (const wxBrush& brush) { @@ -1812,7 +1927,7 @@ void wxPostScriptDC::DoGetTextExtent(const wxString& string, wxString afmName = wxEmptyString; if (!m_printData.GetFontMetricPath().IsEmpty()) { - afmName = m_printData.GetFontMetricPath().fn_str(); + afmName = m_printData.GetFontMetricPath(); } /* 2. open and process the file @@ -1831,29 +1946,31 @@ void wxPostScriptDC::DoGetTextExtent(const wxString& string, /* new elements JC Sun Aug 25 23:21:44 MET DST 1996 */ - afmName << name << ".afm"; - FILE *afmFile = fopen(afmName,"r"); + afmName << name << wxT(".afm"); + FILE *afmFile = wxFopen(afmName,wxT("r")); if (afmFile==NULL) { afmName = wxThePrintSetupData->GetAFMPath(); - afmName << wxFILE_SEP_PATH << name << ".afm"; - afmFile = fopen(afmName,"r"); + afmName << wxFILE_SEP_PATH << name << wxT(".afm"); + afmFile = wxFopen(afmName,wxT("r")); } #ifdef __UNIX__ - if (afmFile==NULL) +#ifndef __VMS__ + if (afmFile==NULL) /* please do NOT change the line above to "else if (afmFile==NULL)" - - afmFile = fopen() may fail and in that case the next if branch MUST be executed - and it would not if there was "else" */ { afmName = wxINSTALL_PREFIX; afmName << wxFILE_SEP_PATH - << "share" << wxFILE_SEP_PATH - << "wx" << wxFILE_SEP_PATH - << "afm" << wxFILE_SEP_PATH - << name << ".afm"; - afmFile = fopen(afmName,"r"); + << wxT("share") << wxFILE_SEP_PATH + << wxT("wx") << wxFILE_SEP_PATH + << wxT("afm") << wxFILE_SEP_PATH + << name << wxT(".afm"); + afmFile = wxFopen(afmName,wxT("r")); } +#endif #endif if (afmFile==NULL) @@ -1876,54 +1993,54 @@ void wxPostScriptDC::DoGetTextExtent(const wxString& string, while(fgets(line,sizeof(line),afmFile)!=NULL) { /* A.) check for descender definition */ - if (wxStrncmp(line,"Descender",9)==0) + if (strncmp(line,"Descender",9)==0) { if ((sscanf(line,"%s%d",descString,&lastDescender)!=2) || - (wxStrcmp(descString,"Descender")!=0)) + (strcmp(descString,"Descender")!=0)) { wxLogDebug( wxT("AFM-file '%hs': line '%hs' has error (bad descender)\n"), afmName.c_str(),line ); } } /* JC 1.) check for UnderlinePosition */ - else if(wxStrncmp(line,"UnderlinePosition",17)==0) + else if(strncmp(line,"UnderlinePosition",17)==0) { if ((sscanf(line,"%s%lf",upString,&UnderlinePosition)!=2) || - (wxStrcmp(upString,"UnderlinePosition")!=0)) + (strcmp(upString,"UnderlinePosition")!=0)) { wxLogDebug( wxT("AFM-file '%hs': line '%hs' has error (bad UnderlinePosition)\n"), afmName.c_str(), line ); } } /* JC 2.) check for UnderlineThickness */ - else if(wxStrncmp(line,"UnderlineThickness",18)==0) + else if(strncmp(line,"UnderlineThickness",18)==0) { if ((sscanf(line,"%s%lf",utString,&UnderlineThickness)!=2) || - (wxStrcmp(utString,"UnderlineThickness")!=0)) + (strcmp(utString,"UnderlineThickness")!=0)) { wxLogDebug( wxT("AFM-file '%hs': line '%hs' has error (bad UnderlineThickness)\n"), afmName.c_str(), line ); } } /* JC 3.) check for EncodingScheme */ - else if(wxStrncmp(line,"EncodingScheme",14)==0) + else if(strncmp(line,"EncodingScheme",14)==0) { if ((sscanf(line,"%s%s",utString,encString)!=2) || - (wxStrcmp(utString,"EncodingScheme")!=0)) + (strcmp(utString,"EncodingScheme")!=0)) { wxLogDebug( wxT("AFM-file '%hs': line '%hs' has error (bad EncodingScheme)\n"), afmName.c_str(), line ); } - else if (wxStrncmp(encString, "AdobeStandardEncoding", 21)) + else if (strncmp(encString, "AdobeStandardEncoding", 21)) { wxLogDebug( wxT("AFM-file '%hs': line '%hs' has error (unsupported EncodingScheme %hs)\n"), afmName.c_str(),line, encString); } } /* B.) check for char-width */ - else if(wxStrncmp(line,"C ",2)==0) + else if(strncmp(line,"C ",2)==0) { if (sscanf(line,"%s%d%s%s%d",cString,&ascii,semiString,WXString,&cWidth)!=5) { wxLogDebug(wxT("AFM-file '%hs': line '%hs' has an error (bad character width)\n"),afmName.c_str(),line); } - if(wxStrcmp(cString,"C")!=0 || wxStrcmp(semiString,";")!=0 || wxStrcmp(WXString,"WX")!=0) + if(strcmp(cString,"C")!=0 || strcmp(semiString,";")!=0 || strcmp(WXString,"WX")!=0) { wxLogDebug(wxT("AFM-file '%hs': line '%hs' has a format error\n"),afmName.c_str(),line); } @@ -1986,10 +2103,12 @@ void wxPostScriptDC::DoGetTextExtent(const wxString& string, } /* add descender to height (it is usually a negative value) */ - if (lastDescender != INT_MIN) - { - height += (wxCoord)(((-lastDescender)/1000.0F) * Size); /* MATTHEW: forgot scale */ - } + //if (lastDescender != INT_MIN) + //{ + // height += (wxCoord)(((-lastDescender)/1000.0F) * Size); /* MATTHEW: forgot scale */ + //} + // - commented by V. Slavik - height already contains descender in it + // (judging from few experiments) /* return size values */ if ( x ) @@ -2037,10 +2156,8 @@ void wxPostScriptDC::DoGetTextExtent(const wxString& string, wxPrintSetupData *wxThePrintSetupData = (wxPrintSetupData *) NULL; -#if !USE_SHARED_LIBRARY IMPLEMENT_DYNAMIC_CLASS(wxPostScriptDC, wxDC) IMPLEMENT_DYNAMIC_CLASS(wxPrintSetupData, wxObject) -#endif // Redundant now I think #if 1