2011-06-02 19:56:07 +00:00
|
|
|
/*
|
|
|
|
* Implementation of Indic Syllables for the Uniscribe Script Processor
|
|
|
|
*
|
|
|
|
* Copyright 2011 CodeWeavers, Aric Stewart
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "winuser.h"
|
|
|
|
#include "wingdi.h"
|
|
|
|
#include "winnls.h"
|
|
|
|
#include "usp10.h"
|
|
|
|
#include "winternl.h"
|
|
|
|
|
|
|
|
#include "wine/debug.h"
|
2018-02-06 00:07:15 +00:00
|
|
|
#include "wine/heap.h"
|
2011-06-02 19:56:07 +00:00
|
|
|
#include "usp10_internal.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(uniscribe);
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static void debug_output_string(const WCHAR *str, unsigned int char_count, lexical_function f)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
if (TRACE_ON(uniscribe))
|
|
|
|
{
|
2017-03-22 22:11:40 +00:00
|
|
|
for (i = 0; i < char_count; ++i)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
switch (f(str[i]))
|
|
|
|
{
|
|
|
|
case lex_Consonant: TRACE("C"); break;
|
|
|
|
case lex_Ra: TRACE("Ra"); break;
|
|
|
|
case lex_Vowel: TRACE("V"); break;
|
|
|
|
case lex_Nukta: TRACE("N"); break;
|
|
|
|
case lex_Halant: TRACE("H"); break;
|
|
|
|
case lex_ZWNJ: TRACE("Zwnj"); break;
|
|
|
|
case lex_ZWJ: TRACE("Zwj"); break;
|
2011-06-06 12:28:54 +00:00
|
|
|
case lex_Matra_post: TRACE("Mp");break;
|
|
|
|
case lex_Matra_above: TRACE("Ma");break;
|
|
|
|
case lex_Matra_below: TRACE("Mb");break;
|
|
|
|
case lex_Matra_pre: TRACE("Mm");break;
|
2011-06-02 19:56:07 +00:00
|
|
|
case lex_Modifier: TRACE("Sm"); break;
|
|
|
|
case lex_Vedic: TRACE("Vd"); break;
|
|
|
|
case lex_Anudatta: TRACE("A"); break;
|
|
|
|
case lex_Composed_Vowel: TRACE("t"); break;
|
|
|
|
default:
|
|
|
|
TRACE("X"); break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TRACE("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-06 12:28:54 +00:00
|
|
|
static inline BOOL is_matra( int type )
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
2011-06-06 12:28:54 +00:00
|
|
|
return (type == lex_Matra_above || type == lex_Matra_below ||
|
2011-10-27 18:38:42 +00:00
|
|
|
type == lex_Matra_pre || type == lex_Matra_post ||
|
|
|
|
type == lex_Composed_Vowel);
|
2011-06-02 19:56:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline BOOL is_joiner( int type )
|
|
|
|
{
|
|
|
|
return (type == lex_ZWJ || type == lex_ZWNJ);
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static int consonant_header(const WCHAR *input, unsigned int cChar,
|
|
|
|
unsigned int start, unsigned int next, lexical_function lex)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
if (!is_consonant( lex(input[next]) )) return -1;
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Nukta)
|
|
|
|
next++;
|
2011-10-19 18:11:19 +00:00
|
|
|
if ((next < cChar) && lex(input[next])==lex_Halant)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
next++;
|
|
|
|
if((next < cChar) && is_joiner( lex(input[next]) ))
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && is_consonant( lex(input[next]) ))
|
|
|
|
return next;
|
|
|
|
}
|
2011-10-19 18:11:19 +00:00
|
|
|
else if ((next < cChar) && is_joiner( lex(input[next]) ) && lex(input[next+1])==lex_Halant)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
next+=2;
|
|
|
|
if ((next < cChar) && is_consonant( lex(input[next]) ))
|
|
|
|
return next;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static int parse_consonant_syllable(const WCHAR *input, unsigned int cChar,
|
|
|
|
unsigned int start, unsigned int *main, unsigned int next, lexical_function lex)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
int check;
|
|
|
|
int headers = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
check = consonant_header(input,cChar,start,next,lex);
|
|
|
|
if (check != -1)
|
|
|
|
{
|
|
|
|
next = check;
|
|
|
|
headers++;
|
|
|
|
}
|
|
|
|
} while (check != -1);
|
|
|
|
if (headers || is_consonant( lex(input[next]) ))
|
|
|
|
{
|
|
|
|
*main = next;
|
|
|
|
next++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Nukta)
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Anudatta)
|
|
|
|
next++;
|
|
|
|
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Halant)
|
|
|
|
{
|
|
|
|
next++;
|
|
|
|
if((next < cChar) && is_joiner( lex(input[next]) ))
|
|
|
|
next++;
|
|
|
|
}
|
|
|
|
else if (next < cChar)
|
|
|
|
{
|
2011-06-06 12:28:54 +00:00
|
|
|
while((next < cChar) && is_matra( lex(input[next]) ))
|
2011-06-02 19:56:07 +00:00
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Nukta)
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Halant)
|
|
|
|
next++;
|
|
|
|
}
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Modifier)
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Vedic)
|
|
|
|
next++;
|
|
|
|
return next;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static int parse_vowel_syllable(const WCHAR *input, unsigned int cChar,
|
|
|
|
unsigned int start, unsigned int next, lexical_function lex)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Nukta)
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && is_joiner( lex(input[next]) ) && lex(input[next+1])==lex_Halant && is_consonant( lex(input[next+2]) ))
|
|
|
|
next+=3;
|
|
|
|
else if ((next < cChar) && lex(input[next])==lex_Halant && is_consonant( lex(input[next+1]) ))
|
|
|
|
next+=2;
|
|
|
|
else if ((next < cChar) && lex(input[next])==lex_ZWJ && is_consonant( lex(input[next+1]) ))
|
|
|
|
next+=2;
|
|
|
|
|
2011-10-19 18:11:19 +00:00
|
|
|
if ((next < cChar) && is_matra( lex(input[next]) ))
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
2011-06-06 12:28:54 +00:00
|
|
|
while((next < cChar) && is_matra( lex(input[next]) ))
|
2011-06-02 19:56:07 +00:00
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Nukta)
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Halant)
|
|
|
|
next++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Modifier)
|
|
|
|
next++;
|
|
|
|
if ((next < cChar) && lex(input[next]) == lex_Vedic)
|
|
|
|
next++;
|
|
|
|
return next;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static int Indic_process_next_syllable(const WCHAR *input, unsigned int cChar,
|
|
|
|
unsigned int start, unsigned int *main, unsigned int next, lexical_function lex)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
if (lex(input[next])==lex_Vowel)
|
|
|
|
{
|
|
|
|
*main = next;
|
|
|
|
return parse_vowel_syllable(input, cChar, start, next+1, lex);
|
|
|
|
}
|
|
|
|
else if ((cChar > next+3) && lex(input[next]) == lex_Ra && lex(input[next+1]) == lex_Halant && lex(input[next+2]) == lex_Vowel)
|
|
|
|
{
|
|
|
|
*main = next+2;
|
|
|
|
return parse_vowel_syllable(input, cChar, start, next+3, lex);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (start == next && lex(input[next])==lex_NBSP)
|
|
|
|
{
|
|
|
|
*main = next;
|
|
|
|
return parse_vowel_syllable(input, cChar, start, next+1, lex);
|
|
|
|
}
|
|
|
|
else if (start == next && (cChar > next+3) && lex(input[next]) == lex_Ra && lex(input[next+1]) == lex_Halant && lex(input[next+2]) == lex_NBSP)
|
|
|
|
{
|
|
|
|
*main = next+2;
|
|
|
|
return parse_vowel_syllable(input, cChar, start, next+3, lex);
|
|
|
|
}
|
|
|
|
|
|
|
|
return parse_consonant_syllable(input, cChar, start, main, next, lex);
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static BOOL Consonant_is_post_base_form(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc,
|
|
|
|
const WCHAR *pwChar, const IndicSyllable *s, lexical_function lexical, BOOL modern)
|
2011-06-15 14:54:07 +00:00
|
|
|
{
|
|
|
|
if (is_consonant(lexical(pwChar[s->base])) && s->base > s->start && lexical(pwChar[s->base-1]) == lex_Halant)
|
2011-06-22 17:38:28 +00:00
|
|
|
{
|
|
|
|
if (modern)
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, &pwChar[s->base-1], 1, 2, "pstf") > 0);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WCHAR cc[2];
|
|
|
|
cc[0] = pwChar[s->base];
|
|
|
|
cc[1] = pwChar[s->base-1];
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, cc, 1, 2, "pstf") > 0);
|
|
|
|
}
|
|
|
|
}
|
2011-06-15 14:54:07 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static BOOL Consonant_is_below_base_form(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc,
|
|
|
|
const WCHAR *pwChar, const IndicSyllable *s, lexical_function lexical, BOOL modern)
|
2011-06-15 14:54:07 +00:00
|
|
|
{
|
|
|
|
if (is_consonant(lexical(pwChar[s->base])) && s->base > s->start && lexical(pwChar[s->base-1]) == lex_Halant)
|
2011-06-22 17:38:28 +00:00
|
|
|
{
|
|
|
|
if (modern)
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, &pwChar[s->base-1], 1, 2, "blwf") > 0);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WCHAR cc[2];
|
|
|
|
cc[0] = pwChar[s->base];
|
|
|
|
cc[1] = pwChar[s->base-1];
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, cc, 1, 2, "blwf") > 0);
|
|
|
|
}
|
|
|
|
}
|
2011-06-15 14:54:07 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static BOOL Consonant_is_pre_base_form(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc,
|
|
|
|
const WCHAR *pwChar, const IndicSyllable *s, lexical_function lexical, BOOL modern)
|
2011-06-15 14:54:07 +00:00
|
|
|
{
|
|
|
|
if (is_consonant(lexical(pwChar[s->base])) && s->base > s->start && lexical(pwChar[s->base-1]) == lex_Halant)
|
2011-06-22 17:38:28 +00:00
|
|
|
{
|
|
|
|
if (modern)
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, &pwChar[s->base-1], 1, 2, "pref") > 0);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WCHAR cc[2];
|
|
|
|
cc[0] = pwChar[s->base];
|
|
|
|
cc[1] = pwChar[s->base-1];
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, cc, 1, 2, "pref") > 0);
|
|
|
|
}
|
|
|
|
}
|
2011-06-15 14:54:07 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static BOOL Consonant_is_ralf(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc,
|
|
|
|
const WCHAR *pwChar, const IndicSyllable *s, lexical_function lexical)
|
2011-06-15 14:54:07 +00:00
|
|
|
{
|
|
|
|
if ((lexical(pwChar[s->start])==lex_Ra) && s->end > s->start && lexical(pwChar[s->start+1]) == lex_Halant)
|
|
|
|
return (SHAPE_does_GSUB_feature_apply_to_chars(hdc, psa, psc, &pwChar[s->start], 1, 2, "rphf") > 0);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
static int FindBaseConsonant(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc,
|
|
|
|
const WCHAR *input, IndicSyllable *s, lexical_function lex, BOOL modern)
|
2011-06-15 14:54:04 +00:00
|
|
|
{
|
|
|
|
int i;
|
2011-06-16 16:28:43 +00:00
|
|
|
BOOL blwf = FALSE;
|
2011-06-16 16:29:03 +00:00
|
|
|
BOOL pref = FALSE;
|
2011-06-15 14:54:07 +00:00
|
|
|
|
|
|
|
/* remove ralf from consideration */
|
2012-05-15 07:47:41 +00:00
|
|
|
if (Consonant_is_ralf(hdc, psa, psc, input, s, lex))
|
2011-06-15 14:54:24 +00:00
|
|
|
{
|
|
|
|
s->ralf = s->start;
|
2011-06-15 14:54:07 +00:00
|
|
|
s->start+=2;
|
2011-06-15 14:54:24 +00:00
|
|
|
}
|
2011-06-15 14:54:07 +00:00
|
|
|
|
2011-06-15 14:54:04 +00:00
|
|
|
/* try to find a base consonant */
|
|
|
|
if (!is_consonant( lex(input[s->base]) ))
|
|
|
|
{
|
|
|
|
for (i = s->end; i >= s->start; i--)
|
|
|
|
if (is_consonant( lex(input[i]) ))
|
|
|
|
{
|
|
|
|
s->base = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-06-15 14:54:07 +00:00
|
|
|
|
2012-05-15 07:47:41 +00:00
|
|
|
while ((blwf = Consonant_is_below_base_form(hdc, psa, psc, input, s, lex, modern)) || Consonant_is_post_base_form(hdc, psa, psc, input, s, lex, modern) || (pref = Consonant_is_pre_base_form(hdc, psa, psc, input, s, lex, modern)))
|
2011-06-15 14:54:07 +00:00
|
|
|
{
|
2011-06-16 16:28:43 +00:00
|
|
|
if (blwf && s->blwf == -1)
|
|
|
|
s->blwf = s->base - 1;
|
2011-06-16 16:29:03 +00:00
|
|
|
if (pref && s->pref == -1)
|
|
|
|
s->pref = s->base - 1;
|
|
|
|
|
2011-06-15 14:54:07 +00:00
|
|
|
for (i = s->base-1; i >= s->start; i--)
|
|
|
|
if (is_consonant( lex(input[i]) ))
|
|
|
|
{
|
|
|
|
s->base = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-15 14:54:24 +00:00
|
|
|
if (s->ralf >= 0)
|
|
|
|
s->start = s->ralf;
|
2011-06-15 14:54:07 +00:00
|
|
|
|
2011-06-16 16:28:43 +00:00
|
|
|
if (s->ralf == s->base)
|
|
|
|
s->ralf = -1;
|
|
|
|
|
2011-06-15 14:54:04 +00:00
|
|
|
return s->base;
|
|
|
|
}
|
|
|
|
|
2017-03-22 22:11:40 +00:00
|
|
|
void Indic_ParseSyllables(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc, const WCHAR *input, unsigned int cChar,
|
|
|
|
IndicSyllable **syllables, int *syllable_count, lexical_function lex, BOOL modern)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
2017-03-22 22:11:40 +00:00
|
|
|
unsigned int center = 0;
|
2011-06-02 19:56:07 +00:00
|
|
|
int index = 0;
|
|
|
|
int next = 0;
|
|
|
|
|
2011-06-15 14:54:00 +00:00
|
|
|
*syllable_count = 0;
|
|
|
|
|
2011-10-27 18:38:42 +00:00
|
|
|
if (!lex)
|
2011-06-02 19:56:07 +00:00
|
|
|
{
|
|
|
|
ERR("Failure to have required functions\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
debug_output_string(input, cChar, lex);
|
|
|
|
while (next != -1)
|
|
|
|
{
|
|
|
|
while((next < cChar) && lex(input[next]) == lex_Generic)
|
|
|
|
next++;
|
|
|
|
index = next;
|
2011-10-14 13:24:58 +00:00
|
|
|
if (next >= cChar)
|
|
|
|
break;
|
2011-06-02 19:56:07 +00:00
|
|
|
next = Indic_process_next_syllable(input, cChar, 0, ¢er, index, lex);
|
|
|
|
if (next != -1)
|
|
|
|
{
|
2011-06-15 14:54:00 +00:00
|
|
|
if (*syllable_count)
|
|
|
|
*syllables = HeapReAlloc(GetProcessHeap(),0,*syllables, sizeof(IndicSyllable)*(*syllable_count+1));
|
|
|
|
else
|
2017-03-09 09:37:07 +00:00
|
|
|
*syllables = heap_alloc(sizeof(**syllables));
|
2011-06-15 14:54:00 +00:00
|
|
|
(*syllables)[*syllable_count].start = index;
|
|
|
|
(*syllables)[*syllable_count].base = center;
|
2011-06-15 14:54:24 +00:00
|
|
|
(*syllables)[*syllable_count].ralf = -1;
|
2011-06-16 16:28:43 +00:00
|
|
|
(*syllables)[*syllable_count].blwf = -1;
|
2011-06-16 16:29:03 +00:00
|
|
|
(*syllables)[*syllable_count].pref = -1;
|
2011-06-15 14:54:00 +00:00
|
|
|
(*syllables)[*syllable_count].end = next-1;
|
2011-06-22 17:38:28 +00:00
|
|
|
FindBaseConsonant(hdc, psa, psc, input, &(*syllables)[*syllable_count], lex, modern);
|
2011-06-02 19:56:07 +00:00
|
|
|
index = next;
|
2011-06-15 14:54:00 +00:00
|
|
|
*syllable_count = (*syllable_count)+1;
|
2011-06-02 19:56:07 +00:00
|
|
|
}
|
|
|
|
else if (index < cChar)
|
|
|
|
{
|
|
|
|
TRACE("Processing failed at %i\n",index);
|
2011-06-27 16:00:58 +00:00
|
|
|
next = ++index;
|
2011-06-02 19:56:07 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-15 14:54:00 +00:00
|
|
|
TRACE("Processed %i of %i characters into %i syllables\n",index,cChar,*syllable_count);
|
2011-06-02 19:56:07 +00:00
|
|
|
}
|
2011-10-27 18:38:42 +00:00
|
|
|
|
2017-03-23 21:49:40 +00:00
|
|
|
void Indic_ReorderCharacters(HDC hdc, SCRIPT_ANALYSIS *psa, ScriptCache *psc, WCHAR *input, unsigned int cChar,
|
|
|
|
IndicSyllable **syllables, int *syllable_count, lexical_function lex, reorder_function reorder_f, BOOL modern)
|
2011-10-27 18:38:42 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!reorder_f)
|
|
|
|
{
|
|
|
|
ERR("Failure to have required functions\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Indic_ParseSyllables(hdc, psa, psc, input, cChar, syllables, syllable_count, lex, modern);
|
|
|
|
for (i = 0; i < *syllable_count; i++)
|
|
|
|
reorder_f(input, &(*syllables)[i], lex);
|
|
|
|
}
|