mirror of
				https://github.com/Rockbox/rockbox.git
				synced 2025-10-24 23:47:38 -04:00 
			
		
		
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@24156 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			297 lines
		
	
	
	
		
			8.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			297 lines
		
	
	
	
		
			8.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2005 by Gadi Cohen
 | |
|  *
 | |
|  * Largely based on php_hebrev by Zeev Suraski <zeev@php.net>
 | |
|  * Heavily modified by Gadi Cohen aka Kinslayer <dragon@wastelands.net>
 | |
|  *
 | |
|  * 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <ctype.h>
 | |
| #include "file.h"
 | |
| #include "lcd.h"
 | |
| #include "rbunicode.h"
 | |
| #include "arabjoin.h"
 | |
| #include "scroll_engine.h"
 | |
| #include "bidi.h"
 | |
| 
 | |
| /* #define _HEB_BUFFER_LENGTH (MAX_PATH + LCD_WIDTH/2 + 3 + 2 + 2) * 2 */
 | |
| #define _HEB_BLOCK_TYPE_ENG 1
 | |
| #define _HEB_BLOCK_TYPE_HEB 0
 | |
| #define _HEB_ORIENTATION_LTR 1
 | |
| #define _HEB_ORIENTATION_RTL 0
 | |
| 
 | |
| #define ischar(c) ((c > 0x0589 && c < 0x0700) || \
 | |
|                    (c >= 0xfb50 && c <= 0xfefc) ? 1 : 0)
 | |
| #define _isblank(c) ((c==' ' || c=='\t') ? 1 : 0)
 | |
| #define _isnewline(c) ((c=='\n' || c=='\r') ? 1 : 0)
 | |
| #define XOR(a,b) ((a||b) && !(a&&b))
 | |
| 
 | |
| #ifndef BOOTLOADER
 | |
| static const arab_t * arab_lookup(unsigned short uchar)
 | |
| {
 | |
|     if (uchar >= 0x621 && uchar <= 0x63a)
 | |
|         return &(jointable[uchar - 0x621]);
 | |
|     if (uchar >= 0x640 && uchar <= 0x64a)
 | |
|         return &(jointable[uchar - 0x621 - 5]);
 | |
|     if (uchar >= 0x671 && uchar <= 0x6d5)
 | |
|         return &(jointable[uchar - 0x621 - 5 - 38]);
 | |
|     if (uchar == 0x200D) /* Support for the zero-width joiner */
 | |
|         return ‍
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static void arabjoin(unsigned short * stringprt, int length)
 | |
| {
 | |
|     bool connected = false;
 | |
|     unsigned short * writeprt = stringprt;
 | |
| 
 | |
|     const arab_t * prev = 0;
 | |
|     const arab_t * cur;
 | |
|     const arab_t * ligature = 0;
 | |
|     short uchar;
 | |
| 
 | |
|     int i;
 | |
|     for (i = 0; i <= length; i++) {
 | |
|         cur = arab_lookup(uchar = *stringprt++);
 | |
| 
 | |
|         /* Skip non-arabic chars */
 | |
|         if (cur == 0) {
 | |
|             if (prev) {
 | |
|                 /* Finish the last char */
 | |
|                 if (connected) {
 | |
|                     *writeprt++ = prev->final;
 | |
|                     connected = false;
 | |
|                 } else
 | |
|                     *writeprt++ = prev->isolated;
 | |
|                 prev = 0;
 | |
|                 *writeprt++ = uchar;
 | |
|             } else {
 | |
|                 *writeprt++ = uchar;
 | |
|             }
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         /* nothing to do for arabic char if the previous was non-arabic */
 | |
|         if (prev == 0) {
 | |
|             prev = cur;
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         /* if it's LAM, check for LAM+ALEPH ligatures */
 | |
|         if (prev->isolated == 0xfedd) {
 | |
|             switch (cur->isolated) {
 | |
|                 case 0xfe8d:
 | |
|                     ligature = &(lamaleph[0]);
 | |
|                     break;
 | |
|                 case 0xfe87:
 | |
|                     ligature = &(lamaleph[1]);
 | |
|                     break;
 | |
|                 case 0xfe83:
 | |
|                     ligature = &(lamaleph[2]);
 | |
|                     break;
 | |
|                 case 0xfe81:
 | |
|                     ligature = &(lamaleph[3]);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (ligature) { /* replace the 2 glyphs by their ligature */
 | |
|             prev = ligature;
 | |
|             ligature = 0;
 | |
|         } else {
 | |
|             if (connected) { /* previous char has something connected to it */
 | |
|                 if (prev->medial && cur->final) /* Can we connect to it? */
 | |
|                     *writeprt++ = prev->medial;
 | |
|                 else {
 | |
|                     *writeprt++ = prev->final;
 | |
|                     connected = false;
 | |
|                 }
 | |
|             } else {
 | |
|                 if (prev->initial && cur->final) { /* Can we connect to it? */
 | |
|                     *writeprt++ = prev->initial;
 | |
|                     connected = true;
 | |
|                 } else
 | |
|                     *writeprt++ = prev->isolated;
 | |
|             }
 | |
|             prev = cur;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| #endif /* !BOOTLOADER */
 | |
| 
 | |
| unsigned short *bidi_l2v(const unsigned char *str, int orientation)
 | |
| {
 | |
|     static unsigned short  utf16_buf[SCROLL_LINE_SIZE];
 | |
|     unsigned short *target, *tmp;
 | |
| #ifndef BOOTLOADER
 | |
|     static unsigned short  bidi_buf[SCROLL_LINE_SIZE];
 | |
|     unsigned short *heb_str; /* *broken_str */
 | |
|     int block_start, block_end, block_type, block_length, i;
 | |
|     int length = utf8length(str);
 | |
| #endif
 | |
|     /*
 | |
|     long max_chars=0;
 | |
|     int begin, end, char_count, orig_begin;
 | |
| 
 | |
|     tmp = str;
 | |
|     */
 | |
|     target = tmp = utf16_buf;
 | |
|     while (*str)
 | |
|         str = utf8decode(str, target++);
 | |
|     *target = 0;
 | |
| 
 | |
| #ifdef BOOTLOADER
 | |
|     (void)orientation;
 | |
|     return utf16_buf;
 | |
|     
 | |
| #else /* !BOOTLOADER */
 | |
|     if (target == utf16_buf) /* empty string */
 | |
|         return target;
 | |
| 
 | |
|     /* properly join any arabic chars */
 | |
|     arabjoin(utf16_buf, length);
 | |
| 
 | |
|     block_start=block_end=block_length=0;
 | |
| 
 | |
|     heb_str = bidi_buf;
 | |
|     if (orientation) {
 | |
|         target = heb_str;
 | |
|     } else {
 | |
|         target = heb_str + length;
 | |
|         *target = 0;
 | |
|         target--;
 | |
|     }
 | |
| 
 | |
|     if (ischar(*tmp))
 | |
|         block_type = _HEB_BLOCK_TYPE_HEB;
 | |
|     else
 | |
|         block_type = _HEB_BLOCK_TYPE_ENG;
 | |
| 
 | |
|     do {
 | |
|         while((XOR(ischar(*(tmp+1)),block_type)
 | |
|                || _isblank(*(tmp+1)) || ispunct((int)*(tmp+1))
 | |
|                || *(tmp+1)=='\n')
 | |
|               && block_end < length-1) {
 | |
|                 tmp++;
 | |
|                 block_end++;
 | |
|                 block_length++;
 | |
|         }
 | |
| 
 | |
|         if (block_type != orientation) {
 | |
|             while ((_isblank(*tmp) || ispunct((int)*tmp))
 | |
|                    && *tmp!='/' && *tmp!='-' && block_end>block_start) {
 | |
|                 tmp--;
 | |
|                 block_end--;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         for (i=block_start; i<=block_end; i++) {
 | |
|             *target = (block_type == orientation) ?
 | |
|                       *(utf16_buf+i) : *(utf16_buf+block_end-i+block_start);
 | |
|             if (block_type!=orientation) {
 | |
|                 switch (*target) {
 | |
|                 case '(':
 | |
|                     *target = ')';
 | |
|                     break;
 | |
|                 case ')':
 | |
|                     *target = '(';
 | |
|                     break;
 | |
|                 default:
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|             target += orientation ? 1 : -1;
 | |
|         }
 | |
|         block_type = !block_type;
 | |
|         block_start=block_end+1;
 | |
|     } while(block_end<length-1);
 | |
| 
 | |
|     *target = 0;
 | |
| 
 | |
| #if 0 /* Is this code really necessary? */
 | |
|     broken_str = utf16_buf;
 | |
|     begin=end=length-1;
 | |
|     target = broken_str;
 | |
| 
 | |
|     while (1) {
 | |
|         char_count=0;
 | |
|         while ((!max_chars || char_count<max_chars) && begin>0) {
 | |
|             char_count++;
 | |
|             begin--;
 | |
|             if (begin<=0 || _isnewline(heb_str[begin])) {
 | |
|                 while(begin>0 && _isnewline(heb_str[begin-1])) {
 | |
|                     begin--;
 | |
|                     char_count++;
 | |
|                 }
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|         if (char_count==max_chars) { /* try to avoid breaking words */
 | |
|             int new_char_count = char_count;
 | |
|             int new_begin = begin;
 | |
|             
 | |
|             while (new_char_count>0) {
 | |
|                 if (_isblank(heb_str[new_begin]) ||
 | |
|                     _isnewline(heb_str[new_begin])) {
 | |
|                     break;
 | |
|                 }
 | |
|                 new_begin++;
 | |
|                 new_char_count--;
 | |
|             }
 | |
|             if (new_char_count>0) {
 | |
|                 char_count=new_char_count;
 | |
|                 begin=new_begin;
 | |
|             }
 | |
|         }
 | |
|         orig_begin=begin;
 | |
|         
 | |
|         /* if (_isblank(heb_str[begin])) {
 | |
|             heb_str[begin]='\n';
 | |
|         } */
 | |
|         
 | |
|         /* skip leading newlines */
 | |
|         while (begin<=end && _isnewline(heb_str[begin])) {
 | |
|             begin++;
 | |
|         }
 | |
| 
 | |
|         /* copy content */
 | |
|         for (i=begin; i<=end; i++) {
 | |
|             *target = heb_str[i];
 | |
|             target++;
 | |
|         }
 | |
| 
 | |
|         for (i=orig_begin; i<=end && _isnewline(heb_str[i]); i++) {
 | |
|             *target = heb_str[i];
 | |
|             target++;
 | |
|         }
 | |
|         begin=orig_begin;
 | |
|         
 | |
|         if (begin<=0) {
 | |
|             *target = 0;
 | |
|             break;
 | |
|         }
 | |
|         begin--;
 | |
|         end=begin;
 | |
|     }
 | |
|     return broken_str;
 | |
| #endif
 | |
|     return heb_str;
 | |
| #endif /* !BOOTLOADER */
 | |
| }
 | |
| 
 |