/* ==================== MissionInformation_Draw ==================== */ void MissionInformation_Draw( centity_t *cent ) { int i,totalY; int iYPixelsPerLine = cgi_R_Font_HeightPixels(cgs.media.qhFontMedium, 1.0f) * (cgi_Language_IsAsian() ? 1.2f : 1.0f ); missionInfo_Updated = qfalse; // This will stop the text from flashing cg.missionInfoFlashTime = 0; // Frame char text[1024]={0}; cgi_SP_GetStringTextString( "INGAME_OBJECTIVES", text, sizeof(text) ); cgi_R_Font_DrawString (96, missionYpos-23, text, colorTable[CT_WHITE], cgs.media.qhFontMedium, -1, 1.0f); int missionYcnt = 0; obj_graphics[0] = obj_graphics[1] = obj_graphics[2] = obj_graphics[3] = qfalse; // Print active objectives cgi_R_SetColor(colorTable[CT_BLUE3]); for (i=0;i<MAX_OBJECTIVES;++i) { if (cent->gent->client->sess.mission_objectives[i].display) { totalY = missionYpos + (iYPixelsPerLine * (missionYcnt))+(iYPixelsPerLine/2); if (obj_graphics[0]) { totalY += 32 + 4; } if (obj_graphics[1]) { totalY += 32 + 4; } if (obj_graphics[2]) { totalY += 32 + 4; } if (obj_graphics[3]) { totalY += 32 + 4; } // OBJECTIVE_STAT_PENDING CG_DrawPic( 88, totalY, 16, 16, cgs.media.messageObjCircle); // Circle in front if (cent->gent->client->sess.mission_objectives[i].status == OBJECTIVE_STAT_SUCCEEDED) { CG_DrawPic( 88, totalY, 16, 16, cgs.media.messageLitOn); // Center Dot } MissionPrint_Line(CT_BLUE3, i, missionYcnt ); } } if (!missionYcnt) { cgi_SP_GetStringTextString( "INGAME_OBJNONE", text, sizeof(text) ); // CG_DrawProportionalString(108, missionYpos, text, CG_SMALLFONT, colorTable[CT_LTBLUE1] ); cgi_R_Font_DrawString (108, missionYpos, text, colorTable[CT_LTBLUE1], cgs.media.qhFontMedium, -1, 1.0f); } }
// display text in a supplied box, start at top left and going down by however many pixels I feel like internally, // return value is NULL if all fitted, else char * of next char to continue from that didn't fit. // // (coords are in the usual 640x480 virtual space)... // // ( if you get the same char * returned as what you passed in, then none of it fitted at all (box too small) ) // const char *CG_DisplayBoxedText(int iBoxX, int iBoxY, int iBoxWidth, int iBoxHeight, const char *psText, int iFontHandle, float fScale, const vec4_t v4Color) { cgi_R_SetColor( v4Color ); // Setup a reasonable vertical spacing (taiwanese & japanese need 1.5 fontheight, so use that for all)... // const int iFontHeight = cgi_R_Font_HeightPixels(iFontHandle, fScale); const int iFontHeightAdvance = (int) (1.5f * (float) iFontHeight); int iYpos = iBoxY; // start print pos // findmeste // test stuff, remove later // psText = "賽卓哥爾博士已經安全了,我也把所有發現報告給「商店」。很不幸地,瑞士警局有些白癡發現了一些狀況,準備在機場逮捕亞歷西•納克瑞得。他偽裝成外交使節,穿過了層層防備。現在他握有人質,並且威脅要散播病毒。根據最搆sCurrentTextReadPos的報告,納克瑞得以及他的黨羽已經完全佔據了機場。我受命來追捕納克瑞得以及救出所有人質。這並不容易。"; // this could probably be simplified now, but it was converted from something else I didn't originally write, // and it works anyway so wtf... // const char *psCurrentTextReadPos = psText; const char *psReadPosAtLineStart = psCurrentTextReadPos; const char *psBestLineBreakSrcPos = psCurrentTextReadPos; const char *psLastGood_s; // needed if we get a full screen of chars with no punctuation or space (see usage notes) while( *psCurrentTextReadPos && (iYpos + iFontHeight < (iBoxY + iBoxHeight)) ) { char sLineForDisplay[2048]; // ott // construct a line... // psCurrentTextReadPos = psReadPosAtLineStart; sLineForDisplay[0] = '\0'; while ( *psCurrentTextReadPos ) { psLastGood_s = psCurrentTextReadPos; // read letter... // qboolean bIsTrailingPunctuation; unsigned int uiLetter = cgi_AnyLanguage_ReadCharFromString(&psCurrentTextReadPos, &bIsTrailingPunctuation); // concat onto string so far... // if (uiLetter == 32 && sLineForDisplay[0] == '\0') { psReadPosAtLineStart++; continue; // unless it's a space at the start of a line, in which case ignore it. } if (uiLetter > 255) { Q_strcat(sLineForDisplay, sizeof(sLineForDisplay),va("%c%c",uiLetter >> 8, uiLetter & 0xFF)); } else { Q_strcat(sLineForDisplay, sizeof(sLineForDisplay),va("%c",uiLetter & 0xFF)); } if (uiLetter == '\n') { // explicit new line... // sLineForDisplay[ strlen(sLineForDisplay)-1 ] = '\0'; // kill the CR psReadPosAtLineStart = psCurrentTextReadPos; psBestLineBreakSrcPos = psCurrentTextReadPos; break; // print this line } else if ( cgi_R_Font_StrLenPixels(sLineForDisplay, iFontHandle, fScale) >= iBoxWidth ) { // reached screen edge, so cap off string at bytepos after last good position... // if (uiLetter > 255 && bIsTrailingPunctuation && !cgi_Language_UsesSpaces()) { // Special case, don't consider line breaking if you're on an asian punctuation char of // a language that doesn't use spaces... // } else { if (psBestLineBreakSrcPos == psReadPosAtLineStart) { // aarrrggh!!!!! we'll only get here is someone has fed in a (probably) garbage string, // since it doesn't have a single space or punctuation mark right the way across one line // of the screen. So far, this has only happened in testing when I hardwired a taiwanese // string into this function while the game was running in english (which should NEVER happen // normally). On the other hand I suppose it'psCurrentTextReadPos entirely possible that some taiwanese string // might have no punctuation at all, so... // psBestLineBreakSrcPos = psLastGood_s; // force a break after last good letter } sLineForDisplay[ psBestLineBreakSrcPos - psReadPosAtLineStart ] = '\0'; psReadPosAtLineStart = psCurrentTextReadPos = psBestLineBreakSrcPos; break; // print this line } } // record last-good linebreak pos... (ie if we've just concat'd a punctuation point (western or asian) or space) // if (bIsTrailingPunctuation || uiLetter == ' ' || (uiLetter > 255 && !cgi_Language_UsesSpaces())) { psBestLineBreakSrcPos = psCurrentTextReadPos; } }
/* ==================== ObjectivePrint_Line Print a single mission objective ==================== */ static void ObjectivePrint_Line(const int color, const int objectIndex, int &missionYcnt) { char *str,*strBegin; int y,pixelLen,charLen,i; const int maxHoldText = 1024; char holdText[maxHoldText]; char finalText[2048]; qhandle_t graphic; int iYPixelsPerLine = cgi_R_Font_HeightPixels(cgs.media.qhFontMedium, 1.0f); cgi_SP_GetStringTextString( va("OBJECTIVES_%s",objectiveTable[objectIndex].name) , finalText, sizeof(finalText) ); // A hack to be able to count prisoners if (objectIndex==T2_RANCOR_OBJ5) { char value[64]; int currTotal, minTotal; gi.Cvar_VariableStringBuffer("ui_prisonerobj_currtotal",value,sizeof(value)); currTotal = atoi(value); gi.Cvar_VariableStringBuffer("ui_prisonerobj_maxtotal",value,sizeof(value)); minTotal = atoi(value); Q_strncpyz(finalText, va(finalText,currTotal,minTotal), sizeof(finalText)); } pixelLen = cgi_R_Font_StrLenPixels(finalText, cgs.media.qhFontMedium, 1.0f); str = finalText; if (cgi_Language_IsAsian()) { // this is execrable, and should NOT have had to've been done now, but... // extern const char *CG_DisplayBoxedText( int iBoxX, int iBoxY, int iBoxWidth, int iBoxHeight, const char *psText, int iFontHandle, float fScale, const vec4_t v4Color); extern int giLinesOutput; extern float gfAdvanceHack; gfAdvanceHack = 1.0f; // override internal vertical advance y = objectiveStartingYpos + (iYPixelsPerLine * missionYcnt); // Advance line if a graphic has printed for (i=0;i<MAX_OBJ_GRAPHICS;i++) { if (obj_graphics[i]) { y += OBJ_GRAPHIC_SIZE + 4; } } CG_DisplayBoxedText( objectiveStartingXpos, y, objectiveTextBoxWidth, objectiveTextBoxHeight, finalText, // int iBoxX, int iBoxY, int iBoxWidth, int iBoxHeight, const char *psText cgs.media.qhFontMedium, // int iFontHandle, 1.0f, // float fScale, colorTable[color] // const vec4_t v4Color ); gfAdvanceHack = 0.0f; // restore missionYcnt += giLinesOutput; } else { // western... // if (pixelLen < objectiveTextBoxWidth) // One shot - small enough to print entirely on one line { y =objectiveStartingYpos + (iYPixelsPerLine * (missionYcnt)); cgi_R_Font_DrawString ( objectiveStartingXpos, y, str, colorTable[color], cgs.media.qhFontMedium, -1, 1.0f); ++missionYcnt; } // Text is too long, break into lines. else { char holdText2[2]; pixelLen = 0; charLen = 0; holdText2[1] = '\0'; strBegin = str; while( *str ) { holdText2[0] = *str; pixelLen += cgi_R_Font_StrLenPixels(holdText2, cgs.media.qhFontMedium, 1.0f); pixelLen += 2; // For kerning ++charLen; if (pixelLen > objectiveTextBoxWidth ) { //Reached max length of this line //step back until we find a space while ((charLen>10) && (*str != ' ' )) { --str; --charLen; } if (*str==' ') { ++str; // To get past space } assert( charLen<maxHoldText ); // Too big? Q_strncpyz( holdText, strBegin, charLen); holdText[charLen] = '\0'; strBegin = str; pixelLen = 0; charLen = 1; y = objectiveStartingYpos + (iYPixelsPerLine * missionYcnt); CG_DrawProportionalString( objectiveStartingXpos, y, holdText, CG_SMALLFONT, colorTable[color] ); ++missionYcnt; } else if (*(str+1) == '\0') { ++charLen; assert( charLen<maxHoldText ); // Too big? y = objectiveStartingYpos + (iYPixelsPerLine * missionYcnt); Q_strncpyz( holdText, strBegin, charLen); CG_DrawProportionalString( objectiveStartingXpos, y, holdText, CG_SMALLFONT, colorTable[color] ); ++missionYcnt; break; } ++str; } } } if (objectIndex == T3_BOUNTY_OBJ1) { y =objectiveStartingYpos + (iYPixelsPerLine * missionYcnt); if (obj_graphics[1]) { y += OBJ_GRAPHIC_SIZE + 4; } if (obj_graphics[2]) { y += OBJ_GRAPHIC_SIZE + 4; } graphic = cgi_R_RegisterShaderNoMip("textures/system/viewscreen1"); CG_DrawPic( 355, 50, OBJ_GRAPHIC_SIZE, OBJ_GRAPHIC_SIZE, graphic ); obj_graphics[3] = qtrue; } }
/* ==================== CG_DrawDataPadObjectives Draw routine for the objective info screen of the data pad. ==================== */ void CG_DrawDataPadObjectives(const centity_t *cent ) { int i,totalY; int iYPixelsPerLine = cgi_R_Font_HeightPixels(cgs.media.qhFontMedium, 1.0f); const short titleXPos = objectiveStartingXpos - 22; // X starting position for title text const short titleYPos = objectiveStartingYpos - 23; // Y starting position for title text const short graphic_size = 16; // Size (width and height) of graphic used to show status of objective const short graphicXpos = objectiveStartingXpos - graphic_size - 8; // Amount of X to backup from text starting position const short graphicYOffset = (iYPixelsPerLine - graphic_size)/2; // Amount of Y to raise graphic so it's in the center of the text line missionInfo_Updated = qfalse; // This will stop the text from flashing cg.missionInfoFlashTime = 0; // zero out objective graphics for (i=0;i<MAX_OBJ_GRAPHICS;i++) { obj_graphics[i] = qfalse; } // Title Text at the top char text[1024]={0}; cgi_SP_GetStringTextString( "SP_INGAME_OBJECTIVES", text, sizeof(text) ); cgi_R_Font_DrawString (titleXPos, titleYPos, text, colorTable[CT_TITLE], cgs.media.qhFontMedium, -1, 1.0f); int missionYcnt = 0; // Print all active objectives for (i=0;i<MAX_OBJECTIVES;i++) { // Is there an objective to see? if (cent->gent->client->sess.mission_objectives[i].display) { // Calculate the Y position totalY = objectiveStartingYpos + (iYPixelsPerLine * (missionYcnt))+(iYPixelsPerLine/2); // Draw graphics that show if mission has been accomplished or not cgi_R_SetColor(colorTable[CT_BLUE3]); CG_DrawPic( (graphicXpos), (totalY-graphicYOffset), graphic_size, graphic_size, cgs.media.messageObjCircle); // Circle in front if (cent->gent->client->sess.mission_objectives[i].status == OBJECTIVE_STAT_SUCCEEDED) { CG_DrawPic( (graphicXpos), (totalY-graphicYOffset), graphic_size, graphic_size, cgs.media.messageLitOn); // Center Dot } // Print current objective text ObjectivePrint_Line(CT_WHITE, i, missionYcnt ); } } // No mission text? if (!missionYcnt) { // Set the message a quarter of the way down and in the center of the text box int messageYPosition = objectiveStartingYpos + (objectiveTextBoxHeight / 4); cgi_SP_GetStringTextString( "SP_INGAME_OBJNONE", text, sizeof(text) ); int messageXPosition = objectiveStartingXpos + (objectiveTextBoxWidth/2) - (cgi_R_Font_StrLenPixels(text, cgs.media.qhFontMedium, 1.0f) /2); cgi_R_Font_DrawString ( messageXPosition, messageYPosition, text, colorTable[CT_WHITE], cgs.media.qhFontMedium, -1, 1.0f); } }
/* ==================== MissionPrint_Line ==================== */ static void MissionPrint_Line(const int color, const int objectIndex, int &missionYcnt) { char *str,*strBegin; int y,pixelLen,charLen; char holdText[1024] ; char finalText[2048]; qhandle_t graphic; int iYPixelsPerLine = cgi_R_Font_HeightPixels(cgs.media.qhFontMedium, 1.0f) * (cgi_Language_IsAsian() ? 1.2f : 1.0f ); #ifndef __NO_JKA if( gi.Cvar_VariableIntegerValue("com_demo") ) { cgi_SP_GetStringTextString( va("OBJECTIVES_DEMO_%s",objectiveTable[objectIndex].name) , finalText, sizeof(finalText) ); } else { cgi_SP_GetStringTextString( va("OBJECTIVES_%s",objectiveTable[objectIndex].name) , finalText, sizeof(finalText) ); } #else cgi_SP_GetStringText( PACKAGE_OBJECTIVES<<8|objectIndex , finalText, sizeof(finalText) ); #endif pixelLen = cgi_R_Font_StrLenPixels(finalText, cgs.media.qhFontMedium, 1.0f); str = finalText; /* CG_DisplayBoxedText(70,50,500,300,finalText, cgs.media.qhFontSmall, 1.0f, colorTable[color] ); */ if (pixelLen < 500) // One shot - small enough to print entirely on one line { y =missionYpos + (iYPixelsPerLine * (missionYcnt)); if (obj_graphics[0]) { y += 32 + 4; } if (obj_graphics[1]) { y += 32 + 4; } if (obj_graphics[2]) { y += 32 + 4; } //CG_DrawProportionalString(108, y,str, CG_SMALLFONT, colorTable[color] ); cgi_R_Font_DrawString (108, y, str, colorTable[color], cgs.media.qhFontMedium, -1, 1.0f); ++missionYcnt; } // Text is too long, break into lines. else { char holdText2[2]; pixelLen = 0; charLen = 0; holdText2[1] = NULL; strBegin = str; while( *str ) { holdText2[0] = *str; pixelLen += cgi_R_Font_StrLenPixels(holdText2, cgs.media.qhFontMedium, 1.0f); pixelLen += 2; // For kerning ++charLen; if (pixelLen > 500 ) { //Reached max length of this line //step back until we find a space while ((charLen) && (*str != ' ' )) { --str; --charLen; } if (*str==' ') { ++str; // To get past space } Q_strncpyz( holdText, strBegin, charLen); holdText[charLen] = NULL; strBegin = str; pixelLen = 0; charLen = 1; y = missionYpos + (iYPixelsPerLine * missionYcnt); CG_DrawProportionalString(108, y, holdText, CG_SMALLFONT, colorTable[color] ); ++missionYcnt; } else if (*(str+1) == NULL) { ++charLen; y = missionYpos + (iYPixelsPerLine * missionYcnt); Q_strncpyz( holdText, strBegin, charLen); CG_DrawProportionalString(108, y, holdText, CG_SMALLFONT, colorTable[color] ); ++missionYcnt; break; } ++str; } } // Special case hack if (objectIndex == DOOM_COMM_OBJ4) { y = missionYpos + (iYPixelsPerLine * missionYcnt); graphic = cgi_R_RegisterShaderNoMip("textures/system/securitycode"); CG_DrawPic( 320 - (128/2), y+8, 128, 32, graphic ); obj_graphics[0] = qtrue; } else if (objectIndex == KEJIM_POST_OBJ3) { y = missionYpos + (iYPixelsPerLine * missionYcnt); graphic = cgi_R_RegisterShaderNoMip("textures/system/securitycode_red"); CG_DrawPic( 320 - (32/2), y+8, 32, 32, graphic ); obj_graphics[1] = qtrue; } else if (objectIndex == KEJIM_POST_OBJ4) { y =missionYpos + (iYPixelsPerLine * missionYcnt); if (obj_graphics[1]) { y += 32 + 4; } graphic = cgi_R_RegisterShaderNoMip("textures/system/securitycode_green"); CG_DrawPic( 320 - (32/2), y+8, 32, 32, graphic ); obj_graphics[2] = qtrue; } else if (objectIndex == KEJIM_POST_OBJ5) { y =missionYpos + (iYPixelsPerLine * missionYcnt); if (obj_graphics[1]) { y += 32 + 4; } if (obj_graphics[2]) { y += 32 + 4; } graphic = cgi_R_RegisterShaderNoMip("textures/system/securitycode_blue"); CG_DrawPic( 320 - (32/2), y+8, 32, 32, graphic ); obj_graphics[3] = qtrue; } }