/* === This file is part of Calamares - === * * SPDX-FileCopyrightText: 2013-2016 Teo Mrnjavac * SPDX-FileCopyrightText: 2018 Adriaan de Groot * SPDX-License-Identifier: GPL-3.0-or-later * * Originally from Tomahawk, portions: * SPDX-FileCopyrightText: 2010-2011 Christian Muehlhaeuser * SPDX-FileCopyrightText: 2010-2011 Leo Franchi * SPDX-FileCopyrightText: 2010-2012 Jeff Mitchell * * Calamares is Free Software: see the License-Identifier above. * * */ #include "String.h" #include "Logger.h" #include namespace Calamares { namespace String { QString removeDiacritics( const QString& string ) { // *INDENT-OFF* // clang-format off // Map these extended-Latin characters to ASCII; keep the // layout so that one line in *diacriticLetters* corresponds // to one line of replacements in *noDiacriticLetters*. static const QString diacriticLetters = QString::fromUtf8( "ŠŒŽšœžŸ¥µÀ" "ÁÂÃÄÅÆÇÈÉÊ" "ËÌÍÎÏÐÑÒÓÔ" "ÕÖØÙÚÛÜÝßà" "áâãäåæçèéê" "ëìíîïðñòóô" "õöøùúûüýÿÞ" "þČčĆćĐ𩹮" "žŞşĞğİıȚțȘ" "șĂăŐőŰűŘřĀ" "āĒēĪīŌōŪūŢ" "ţẀẁẂẃŴŵŶŷĎ" "ďĚěŇňŤťŮůŔ" "ॹĘꣳŃńŚ" "śŹźŻż" ); static const QStringList noDiacriticLetters = { "S", "OE", "Z", "s", "oe", "z", "Y", "Y", "u", "A", "A", "A", "A", "A", "AA", "AE", "C", "E", "E", "E", "E", "I", "I", "I", "I", "D", "N", "O", "O", "O", "O", "E", "OE", "U", "U", "U", "E", "Y", "s", "a", "a", "a", "a", "e", "aa", "ae", "c", "e", "e", "e", "e", "i", "i", "i", "i", "d", "n", "o", "o", "o", "o", "e", "oe", "u", "u", "u", "e", "y", "y", "TH", "th", "C", "c", "C", "c", "DJ", "dj", "S", "s", "Z", "z", "S", "s", "G", "g", "I", "i", "T", "t", "S", "s", "A", "a", "O", "o", "U", "u", "R", "r", "A", "a", "E", "e", "I", "i", "O", "o", "U", "u", "T", "t", "W", "w", "W", "w", "W", "w", "Y", "y", "D", "d", "E", "e", "N", "n", "T", "t", "U", "u", "R", "r", "A", "a", "E", "e", "L", "l", "N", "n", "S", "s", "Z", "z", "Z", "z" }; // clang-format on // *INDENT-ON* QString output; for ( const QChar& c : string ) { int i = diacriticLetters.indexOf( c ); if ( i < 0 ) { output.append( c ); } else { QString replacement = noDiacriticLetters[ i ]; output.append( replacement ); } } return output; } // Function Calamares::obscure based on KStringHandler::obscure, // part of KDElibs by KDE, file kstringhandler.cpp. // Original copyright statement follows. /* This file is part of the KDE libraries Copyright (C) 1999 Ian Zepp (icszepp@islc.net) Copyright (C) 2006 by Dominic Battre Copyright (C) 2006 by Martin Pool This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ QString obscure( const QString& string ) { QString result; const QChar* unicode = string.unicode(); for ( int i = 0; i < string.length(); ++i ) // yes, no typo. can't encode ' ' or '!' because // they're the unicode BOM. stupid scrambling. stupid. { result += ( unicode[ i ].unicode() <= 0x21 ) ? unicode[ i ] : QChar( 0x1001F - unicode[ i ].unicode() ); } return result; } QString truncateMultiLine( const QString& string, LinesStartEnd lines, CharCount chars ) { const char NEWLINE = '\n'; const int maxLines = lines.atStart + lines.atEnd; if ( maxLines < 1 ) { QString shorter( string ); shorter.truncate( chars.total ); return shorter; } const int physicalLinesInString = string.count( NEWLINE ); const int logicalLinesInString = physicalLinesInString + ( string.endsWith( NEWLINE ) ? 0 : 1 ); if ( ( string.length() <= chars.total ) && ( logicalLinesInString <= maxLines ) ) { return string; } QString front, back; if ( physicalLinesInString >= maxLines ) { int from = -1; for ( int i = 0; i < lines.atStart; ++i ) { from = string.indexOf( NEWLINE, from + 1 ); if ( from < 0 ) { // That's strange, we counted at least maxLines newlines before break; } } if ( from > 0 ) { front = string.left( from + 1 ); } int lastNewLine = -1; int lastCount = string.endsWith( NEWLINE ) ? -1 : 0; for ( auto i = string.rbegin(); i != string.rend() && lastCount < lines.atEnd; ++i ) { if ( *i == NEWLINE ) { ++lastCount; lastNewLine = int( i - string.rbegin() ); } } if ( ( lastNewLine >= 0 ) && ( lastCount >= lines.atEnd ) ) { back = string.right( lastNewLine ); } } else { // We have: <= maxLines and longer than chars.total, so: // - carve out a chunk in the middle, based a little on // how the balance of atStart and atEnd is const int charsToChop = string.length() - chars.total; if ( charsToChop < 1 ) { // That's strange, again return string; } const int startPortion = charsToChop * lines.atStart / maxLines; const int endPortion = charsToChop * lines.atEnd / maxLines; front = string.left( string.length() / 2 - startPortion ); back = string.right( string.length() / 2 - endPortion ); } if ( front.length() + back.length() <= chars.total ) { return front + back; } // We need to cut off some bits, preserving whether there are // newlines present at the end of the string. Go case-by-case: if ( !front.isEmpty() && back.isEmpty() ) { // Truncate towards the front bool needsNewline = front.endsWith( NEWLINE ); front.truncate( chars.total ); if ( !front.endsWith( NEWLINE ) && needsNewline ) { front.append( NEWLINE ); } return front; } if ( front.isEmpty() && !back.isEmpty() ) { // Truncate towards the tail return back.right( chars.total ); } // Both are non-empty, so nibble away at both of them front.truncate( chars.total / 2 ); if ( !front.endsWith( NEWLINE ) && physicalLinesInString > 0 ) { front.append( NEWLINE ); } return front + back.right( chars.total / 2 ); } void removeLeading( QString& string, QChar c ) { int count = 0; while ( string.length() > count && string[ count ] == c ) { count++; } string.remove( 0, count ); } void removeTrailing( QString& string, QChar c ) { int lastIndex = string.length(); while ( lastIndex > 0 && string[ lastIndex - 1 ] == c ) { lastIndex--; } string.remove( lastIndex, string.length() ); } } // namespace String } // namespace Calamares