mirror of
https://github.com/libretro/scummvm.git
synced 2024-12-14 13:50:13 +00:00
db8559ff3e
Fixes bug #12337
148 lines
4.7 KiB
C++
148 lines
4.7 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
#include "common/ustr.h"
|
|
#include "common/unicode-bidi.h"
|
|
#include "common/textconsole.h"
|
|
|
|
#ifdef USE_FRIBIDI
|
|
#include <fribidi/fribidi.h>
|
|
#else
|
|
/* This constant is used below in common code
|
|
* fake it here to lighten code
|
|
*/
|
|
#define FRIBIDI_PAR_ON 0
|
|
#endif
|
|
|
|
namespace Common {
|
|
|
|
UnicodeBiDiText::UnicodeBiDiText(const Common::U32String &str) :
|
|
logical(str), _pbase_dir(FRIBIDI_PAR_ON),
|
|
_log_to_vis_index(NULL), _vis_to_log_index(NULL) {
|
|
initWithU32String(str);
|
|
}
|
|
|
|
UnicodeBiDiText::UnicodeBiDiText(const Common::String &str, const Common::CodePage page) :
|
|
logical(str), _pbase_dir(FRIBIDI_PAR_ON),
|
|
_log_to_vis_index(NULL), _vis_to_log_index(NULL) {
|
|
initWithU32String(str.decode(page));
|
|
}
|
|
|
|
UnicodeBiDiText::UnicodeBiDiText(const Common::String &str, const Common::CodePage page,
|
|
uint32 *pbase_dir) : logical(str), _log_to_vis_index(NULL), _vis_to_log_index(NULL) {
|
|
_pbase_dir = *pbase_dir;
|
|
initWithU32String(str.decode(page));
|
|
*pbase_dir = _pbase_dir;
|
|
}
|
|
|
|
UnicodeBiDiText::~UnicodeBiDiText() {
|
|
delete[] _log_to_vis_index;
|
|
delete[] _vis_to_log_index;
|
|
}
|
|
|
|
uint32 UnicodeBiDiText::getVisualPosition(uint32 logicalPos) const {
|
|
if (NULL != _log_to_vis_index && logicalPos < size()) {
|
|
return _log_to_vis_index[logicalPos];
|
|
}
|
|
return logicalPos;
|
|
}
|
|
uint32 UnicodeBiDiText::getLogicalPosition(uint32 visualPos) const {
|
|
if (NULL != _log_to_vis_index && visualPos < size()) {
|
|
return _vis_to_log_index[visualPos];
|
|
}
|
|
return visualPos;
|
|
}
|
|
|
|
void UnicodeBiDiText::initWithU32String(const U32String &input) {
|
|
|
|
#ifdef USE_FRIBIDI
|
|
uint32 input_size = input.size();
|
|
uint32 buff_length = (input_size + 2) * 2; // it's more than enough, but it's better to be on the safe side
|
|
FriBidiChar *visual_str = new FriBidiChar[buff_length * sizeof(FriBidiChar)];
|
|
_log_to_vis_index = new uint32[input_size];
|
|
_vis_to_log_index = new uint32[input_size];
|
|
|
|
FriBidiParType pbase_dir = FriBidiParType(_pbase_dir);
|
|
if (!fribidi_log2vis(
|
|
/* input */
|
|
(const FriBidiChar *)input.c_str(),
|
|
input_size,
|
|
&pbase_dir,
|
|
/* output */
|
|
visual_str,
|
|
(FriBidiStrIndex *)_log_to_vis_index, // position_L_to_V_list,
|
|
(FriBidiStrIndex *)_vis_to_log_index, // position_V_to_L_list,
|
|
NULL // embedding_level_list
|
|
)) {
|
|
warning("initWithU32String: calling fribidi_log2vis failed");
|
|
delete[] visual_str;
|
|
delete[] _log_to_vis_index;
|
|
delete[] _vis_to_log_index;
|
|
visual = input;
|
|
_log_to_vis_index = NULL;
|
|
_vis_to_log_index = NULL;
|
|
} else {
|
|
visual = U32String((uint32 *)visual_str, input.size());
|
|
delete[] visual_str;
|
|
}
|
|
_pbase_dir = pbase_dir;
|
|
#else
|
|
static bool fribidiWarning = true;
|
|
if (fribidiWarning) {
|
|
warning("initWithU32String: Fribidi not available, will use input strings as fallback.");
|
|
fribidiWarning = false;
|
|
}
|
|
visual = input;
|
|
#endif
|
|
|
|
}
|
|
|
|
Common::String bidiByLineHelper(Common::String line, va_list args) {
|
|
/* Common::CodePage is int
|
|
* GCC warns that using Common::CodePage in va_arg would abort program */
|
|
Common::CodePage page = (Common::CodePage) va_arg(args, int);
|
|
uint32 *pbase_dir = va_arg(args, uint32*);
|
|
return UnicodeBiDiText(line, page, pbase_dir).visual.encode(page);
|
|
}
|
|
|
|
String convertBiDiStringByLines(const String &input, const Common::CodePage page) {
|
|
uint32 pbase_dir = FRIBIDI_PAR_ON;
|
|
return input.forEachLine(bidiByLineHelper, page, &pbase_dir);
|
|
}
|
|
|
|
String convertBiDiString(const String &input, const Common::Language lang) {
|
|
if (lang != Common::HE_ISR) //TODO: modify when we'll support other RTL languages, such as Arabic and Farsi
|
|
return input;
|
|
|
|
return Common::convertBiDiString(input, kWindows1255);
|
|
}
|
|
|
|
String convertBiDiString(const String &input, const Common::CodePage page) {
|
|
return convertBiDiU32String(input.decode(page)).visual.encode(page);
|
|
}
|
|
|
|
UnicodeBiDiText convertBiDiU32String(const U32String &input) {
|
|
return UnicodeBiDiText(input);
|
|
}
|
|
|
|
} // End of namespace Common
|