CrazyPhD 3 years ago
commit
a5af4012bb

+ 35 - 0
index.html

@@ -0,0 +1,35 @@
+<!DOCTYPE html>
+<html>
+	<head>
+		<title>Поиск палиндромов</title>
+		<meta name="viewport" content="width=device-width, initial-scale=1" /> 
+		<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
+		<link rel="stylesheet" href="./style/fonts.css" />
+		<link rel="stylesheet" href="./style/style.css" />
+		<link rel="stylesheet" href="./style/mobile.css" />
+		<script src="./js/process.js"></script>
+	</head>
+	<body>
+		<div class="header">
+			<div class="header__logo">
+				<div class="logo__m"></div>
+			</div>
+		</div>
+		<div class="body">
+			<form name="process_form" class="body__form">
+				<div class="form__title">
+					<span>Приложение</span>
+				</div>
+				<div class="form__subtitle">
+					<span>Веб-приложение, которое находит палиндромы</span>
+				</div>
+				<input id="phrase" class="form__input" type="text" name="phrase" placeholder="палиндром" />
+				<input id="find" class="form__btn" type="submit" name="find" value="Найти" />
+				<div id="result" class="body__result hidden">
+					<div id="result_summary" class="result__summary"></div>
+					<ol id="result_list" class="result__list"></ol>
+				</div>
+			</form>
+		</div>
+	</body>
+</html>

+ 45 - 0
js/process.js

@@ -0,0 +1,45 @@
+'use strict';
+
+window.addEventListener('click', (e) => {
+	if (e.target.id === 'find') {
+		getWords();
+		e.preventDefault();
+	}
+});
+
+function getWords() {
+	const xhr = new XMLHttpRequest();
+	const formData = new FormData(document.forms.process_form);
+	const resultsContainer = document.getElementById("result");
+	const resultsVerdict = document.getElementById("result_summary");
+	const resultsList = document.getElementById("result_list");
+	
+	xhr.open("POST", "src/get_words_list.php");
+	xhr.onload = function (e) {
+		if (xhr.readyState == 4 && xhr.status == 200) {
+			
+			/* Clean previous results if they were shown */
+			resultsVerdict.innerHTML = "";
+			resultsList.innerHTML = "";
+			
+			/* Get response and decode it */
+			let response = JSON.parse(xhr.response);
+			
+			/* Get summary message from response */
+			resultsVerdict.innerHTML = response.verdict;
+			
+			/* Fill results list with found words */
+			if (response.count > 0) {
+				for (let word of response.words) {
+					let wordListElement = document.createElement("li");
+					wordListElement.innerHTML = word;
+					resultsList.appendChild(wordListElement);
+				}
+			}
+			
+			/* If results container is hidden, show it */
+			resultsContainer.classList.remove("hidden");
+		}
+	};
+	xhr.send(formData);
+}

+ 26 - 0
src/get_words_list.php

@@ -0,0 +1,26 @@
+<?php
+	include_once "palindrome.php"; /* Words filter logic source */
+	
+	$palindrome = new Palindrome();
+	
+	/* Words filter logic function */
+	$algorithm = $palindrome->subpalindromes;
+	
+	/*	If something is found, success message
+		Note:	It is assumed, that it is an anonymous function with number as parameter.
+				If it is necessary, change its usage below in desired way. ("verdict" field in json output)
+	*/
+	$verdict_success = $palindrome->success;
+	
+	/* Nothing is found, failure message */
+	$verdict_fail = $palindrome->fail;
+	
+	if (isset($_POST['phrase'])) {
+		$foundWords = $algorithm($_POST['phrase']);
+		$foundWordsCount = count($foundWords);
+		print json_encode(array(
+					"count" => $foundWordsCount,
+					"words" => $foundWords,
+					"verdict" => ($foundWordsCount > 0 ? $verdict_success($foundWordsCount) : $verdict_fail)
+					));
+	}

+ 113 - 0
src/palindrome.bak.php

@@ -0,0 +1,113 @@
+<?php
+
+	/*	Algorithm for finding (sub)palindromes in phrase
+		Note: if there are spaces, they will be ignored
+		and the string will be treated as if they weren't there.
+		Also, all duplicates will not be included in result.
+		
+		Manacher's Algorithm used here.
+	*/
+	
+	mb_internal_encoding("UTF-8"); // UTF-8 support
+	
+	$palindrome = function ($phrase) {
+		
+		$phrase = mb_strtolower($phrase); // mb_* functions used to support UTF-8 encoding.
+		$phrase = mb_ereg_replace(" ", "", $phrase); // remove all spaces
+		
+		
+		$subpalindromes = manacher($phrase); // run Manacher's algorithm to find odd and even subpalindromes
+		$odd_subpalindromes = $subpalindromes[0];
+		$even_subpalindromes = $subpalindromes[1];
+		
+		$palindromes = [];
+		for ($i = 0; $i < mb_strlen($phrase); $i++) {
+			if ($odd_subpalindromes[$i] > 1) { // ignore palindromes which consits of one character
+				$palindromeLength = $odd_subpalindromes[$i]; // $odd_subpalindromes represents array of lengths of biggest palindromes, where index is the index of the character, which is the center of this palindrome
+				while ($palindromeLength > 1) { 
+					$palindromes[] = mb_substr($phrase, $i - $palindromeLength + 1, $palindromeLength*2 - 1);
+					$palindromeLength--;
+				}
+			}
+			// same as previous with specific changes
+			if ($even_subpalindromes[$i] > 0) {
+				$palindromeLength = $even_subpalindromes[$i];
+				while ($palindromeLength > 0) {
+					$palindromes[] = mb_substr($phrase, $i - $palindromeLength, $palindromeLength*2);
+					$palindromeLength--;
+				}
+			}
+		}
+		return array_values(array_unique($palindromes));
+	};
+	
+	/*	Manacher's algorithm. 
+		Time complexity: O(n)
+		Space complexity: O(1)
+	*/
+	function manacher($str) {
+		$n = mb_strlen($str);
+		$odd = array_fill(0, $n, 0);
+		$even = array_fill(0, $n, 0);
+		
+		/* Odd subpalindromes */
+		$l = 0; // left edge of current rightest palindrome
+		$r = -1; // right edge of current rightst palindrome
+		for ($i = 0; $i < $n; $i++) {	
+			$k = $i > $r ? 1 : min($odd[$l + $r - $i], $r - $i + 1); // k - known via previous steps palindrome center offset (guaranteed biggest palindrome with center in current position [i-k, i+k]
+			while ($i + $k < $n && $i - $k >= 0 && mb_substr($str, $i - $k, 1) == mb_substr($str, $i + $k, 1)) {
+				$k++; // increment offset if we are still between 0 and string length and mirrored elements are equal
+			}
+			$odd[$i] = $k; // here we know max palindrome size with center in current position (i)
+			if ($i + $k - 1 > $r) { // if right edge of current palindrome is righter than right edge of the previous rightest palindrome, renew l and r 
+				$l = $i - $k + 1;
+				$r = $i + $k - 1;
+			}
+		}
+		
+		/*	Even subpalindromes
+			a bit modified previous algorithm (for odd subpalindromes)
+		*/
+		$l = 0;
+		$r = -1;
+		for ($i = 0; $i < $n; $i++) {
+			$k = $i > $r ? 0 : min($even[$l + $r - $i + 1], $r - $i + 1);
+			while ($i + $k < $n && $i - $k - 1 >= 0 && mb_substr($str, $i + $k, 1) == mb_substr($str, $i - $k - 1, 1)) {
+				$k++;
+			}
+			$even[$i] = $k;
+			if ($i + $k - 1 > $r) {
+				$l = $i - $k;
+				$r = $i + $k - 1;
+			}
+		}
+		
+		return [$odd, $even];
+	}
+	
+	/*	Used for formatting correct success message
+		depending on the number of words found.
+	*/
+	$success_message = function ($n) {
+		$n_1 = $n % 100;
+		
+		if ($n_1 > 19) {
+			$n_1 = $n_1 % 10;
+		}
+		
+		$palindromeWordForm = "палиндром";
+		$foundWordForm = "Найдено";
+		
+		if ($n_1 >= 2 && $n_1 <= 4) {
+			$palindromeWordForm = "палиндрома";
+		} else if ($n_1 == 1) {
+			$palindromeWordForm = "палиндром";
+			$foundWordForm = "Найден";
+		} else {
+			$palindromeWordForm = "палиндромов";
+		}
+		
+		return $foundWordForm." ".strval($n)." ".$palindromeWordForm;
+	};
+	
+	$fail_message = "Не найдено ни одного палиндрома";

+ 124 - 0
src/palindrome.php

@@ -0,0 +1,124 @@
+<?php
+
+	/*	Algorithm for finding (sub)palindromes in phrase
+		Note: if there are spaces, they will be ignored
+		and the string will be treated as if they weren't there.
+		Also, all duplicates will not be included in result.
+		
+		Manacher's Algorithm used here.
+	*/
+	
+	mb_internal_encoding("UTF-8"); // UTF-8 support
+		
+	class Palindrome {
+		
+		public $subpalindromes;
+		public $success;
+		public $fail;
+		
+		function __construct() {
+			$this->subpalindromes = function ($phrase) {
+				$phrase = mb_strtolower($phrase); // mb_* functions used to support UTF-8 encoding.
+				$phrase = mb_ereg_replace(" ", "", $phrase); // remove all spaces
+				
+				
+				$manacher_subpalindromes = self::manacher($phrase); // run Manacher's algorithm to find odd and even subpalindromes
+				$odd_subpalindromes = $manacher_subpalindromes[0];
+				$even_subpalindromes = $manacher_subpalindromes[1];
+				
+				$palindromes = [];
+				for ($i = 0; $i < mb_strlen($phrase); $i++) {
+					if ($odd_subpalindromes[$i] > 1) { // ignore palindromes which consits of one character
+						$palindromeLength = $odd_subpalindromes[$i]; // $odd_subpalindromes represents array of lengths of biggest palindromes, where index is the index of the character, which is the center of this palindrome
+						while ($palindromeLength > 1) { 
+							$palindromes[] = mb_substr($phrase, $i - $palindromeLength + 1, $palindromeLength*2 - 1);
+							$palindromeLength--;
+						}
+					}
+					
+					// same as previous with specific changes
+					if ($even_subpalindromes[$i] > 0) {
+						$palindromeLength = $even_subpalindromes[$i];
+						while ($palindromeLength > 0) {
+							$palindromes[] = mb_substr($phrase, $i - $palindromeLength, $palindromeLength*2);
+							$palindromeLength--;
+						}
+					}
+				}
+				return array_values(array_unique($palindromes));
+			};
+			
+			/*	Used for formatting correct success message
+				depending on the number of words found.
+			*/
+			$this->success = function ($n) {
+				$n_1 = $n % 100;
+			
+				if ($n_1 > 19) {
+					$n_1 = $n_1 % 10;
+				}
+				
+				$palindromeWordForm = "палиндром";
+				$foundWordForm = "Найдено";
+				
+				if ($n_1 >= 2 && $n_1 <= 4) {
+					$palindromeWordForm = "палиндрома";
+				} else if ($n_1 == 1) {
+					$palindromeWordForm = "палиндром";
+					$foundWordForm = "Найден";
+				} else {
+					$palindromeWordForm = "палиндромов";
+				}
+				
+				return $foundWordForm." ".strval($n)." ".$palindromeWordForm;
+			};
+			
+			$this->fail = "Не найдено ни одного палиндрома";
+		}
+		
+		
+		
+		/*	Manacher's algorithm. 
+			Time complexity: O(n)
+			Space complexity: O(1)
+		*/		
+		private static function manacher($str) {
+			$n = mb_strlen($str);
+			$odd = array_fill(0, $n, 0); // odd subpalindromes
+			$even = array_fill(0, $n, 0); // even subpalindromes
+			
+			/* Odd subpalindromes */
+			$l = 0; // left edge of current rightest palindrome
+			$r = -1; // right edge of current rightst palindrome
+			for ($i = 0; $i < $n; $i++) {	
+				$k = $i > $r ? 1 : min($odd[$l + $r - $i], $r - $i + 1); // k - known via previous steps palindrome center offset (guaranteed biggest palindrome with center in current position [i-k, i+k]
+				while ($i + $k < $n && $i - $k >= 0 && mb_substr($str, $i - $k, 1) == mb_substr($str, $i + $k, 1)) {
+					$k++; // increment offset if we are still between 0 and string length and mirrored elements are equal
+				}
+				$odd[$i] = $k; // here we know max palindrome size with center in current position (i)
+				if ($i + $k - 1 > $r) { // if right edge of current palindrome is righter than right edge of the previous rightest palindrome, renew l and r 
+					$l = $i - $k + 1;
+					$r = $i + $k - 1;
+				}
+			}
+			
+			/*	Even subpalindromes
+				a bit modified previous algorithm (for odd subpalindromes)
+			*/
+			$l = 0;
+			$r = -1;
+			for ($i = 0; $i < $n; $i++) {
+				$k = $i > $r ? 0 : min($even[$l + $r - $i + 1], $r - $i + 1);
+				while ($i + $k < $n && $i - $k - 1 >= 0 && mb_substr($str, $i + $k, 1) == mb_substr($str, $i - $k - 1, 1)) {
+					$k++;
+				}
+				$even[$i] = $k;
+				if ($i + $k - 1 > $r) {
+					$l = $i - $k;
+					$r = $i + $k - 1;
+				}
+			}
+			
+			return [$odd, $even];
+		}
+	}

BIN
style/font/Montserrat-Black.ttf


BIN
style/font/Montserrat-Bold.ttf


BIN
style/font/Montserrat-ExtraBold.ttf


BIN
style/font/Montserrat-Light.ttf


BIN
style/font/Montserrat-Regular.ttf


+ 24 - 0
style/fonts.css

@@ -0,0 +1,24 @@
+@font-face {
+	font-family: "Montserrat Regular";
+	src: url("./font/Montserrat-Regular.ttf") format("truetype");
+}
+
+@font-face {
+	font-family: "Montserrat Light";
+	src: url("./font/Montserrat-Light.ttf") format("truetype");
+}
+
+@font-face {
+	font-family: "Montserrat Black";
+	src: url("./font/Montserrat-Black.ttf") format("truetype");
+}
+
+@font-face {
+	font-family: "Montserrat Bold";
+	src: url("./font/Montserrat-Bold.ttf") format("truetype");
+}
+
+@font-face {
+	font-family: "Montserrat ExtraBold";
+	src: url("./font/Montserrat-ExtraBold.ttf") format("truetype");
+}

+ 17 - 0
style/mobile.css

@@ -0,0 +1,17 @@
+/* For most of the mobile devices. */
+@media (max-width: 768px) {
+	.header__logo {
+		left: 15px;
+	}
+	
+	.body__form {
+		width: 320px;
+	}
+}
+
+/* Galaxy Fold etc. width <= 330 */
+@media (max-width: 330px) {
+	.body__form {
+		width: 270px;
+	}
+}

+ 138 - 0
style/style.css

@@ -0,0 +1,138 @@
+body {
+	height: 100vh;
+	margin: 0;
+}
+
+.header {
+	width: 100%;
+	height: 55px;
+	background: #FFFFFF 0% 0% no-repeat padding-box;
+	box-shadow: 0px 1px 0px #0000001A;
+}
+
+.header__logo {
+	position: relative;
+	left: 90px;
+	width: 60px;
+	height: 56px;
+	background: #1082D2 0% 0% no-repeat padding-box;
+}
+
+.logo__m {
+	height: 56px;
+	line-height: 56px;
+	text-align: center;
+}
+
+.logo__m:before {
+	font-family: "Montserrat Bold";
+	font-size: 34px;
+	color: #FFFFFF;
+	content: 'M';
+}
+
+.body {
+	width: 100%;
+	position: relative;
+	height: calc(100vh - 57px);
+}
+
+.body__form {
+	height: fit-content;
+    width: 330px;
+    margin: auto;
+    top: 0;
+    bottom: 0;
+    left: 0;
+    right: 0;
+    position: absolute;
+}
+.form__title {
+    font: normal normal normal 30px/24px "Montserrat Bold";
+    letter-spacing: -1px;
+    color: #000000;
+    height: 35px;
+    display: block;
+}
+
+.form__subtitle {
+	font: normal normal normal 14px/18px "Montserrat Regular";
+    letter-spacing: -0.8px;
+	margin-bottom: 20px;
+}
+
+.form__input {
+	height: 40px;
+	width: calc(100% - 22px);
+	background: #FFFFFF 0% 0% no-repeat padding-box;
+	border: 1px solid #ECECEC;
+	padding: 0 10px;
+	display: block;
+	font: normal normal normal 14px/15px "Montserrat Regular";
+	letter-spacing: -0.28px;
+	margin-bottom: 20px;
+}
+
+.form__input:focus {
+    outline:none;
+}
+
+.form__input::placeholder {
+	color: #CCC9C9;
+}
+
+.form__btn {
+	height: 42px;
+	width: 100%;
+	background: #1082D2 0% 0% no-repeat padding-box;
+	font: normal normal bold 15px/23px "Montserrat Bold";
+	letter-spacing: 0px;
+	color: #FFFFFF;
+	border: 0;
+	padding: 0;
+	display: block;
+	margin-bottom: 50px;
+}
+
+.form__btn:focus {
+    outline:none;
+}
+
+.form__btn:hover {
+	cursor: pointer;
+	background: #0d79c5 0% 0% no-repeat padding-box;
+}
+
+.result__summary, .result__list {
+	font: normal normal normal 14px/18px "Montserrat Regular";
+    letter-spacing: -0.8px;
+}
+
+.body__result {
+	height: fit-content;
+	max-height: 500px;
+	overflow-x: hidden;
+    overflow-y: auto;
+	transition: max-height 1s;
+}
+
+.hidden {
+	max-height: 0;
+	transition: max-height 1s;
+}
+
+.result__summary:after {
+	content: '';
+	border-bottom: 1px solid #000000;
+	display: block;
+	width: 100%;
+	height: 8.5px;
+	opacity: 0.15;
+	margin-bottom: 29.5px
+}
+
+.result__list {
+	list-style-position: inside;
+	padding: 0;
+	margin: 0;
+}