mirror of
				https://github.com/Mueller-Patrick/Betterzon.git
				synced 2025-10-25 22:05:48 +00:00 
			
		
		
		
	Compare commits
	
		
			5 Commits
		
	
	
		
			b839994de0
			...
			1b57ebfb53
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1b57ebfb53 | ||
|  | ddebe78f15 | ||
| 086e0d41b3 | |||
| 2bc97503de | |||
| 8049fd9987 | 
|  | @ -3,22 +3,29 @@ package stepdefs; | ||||||
| import io.cucumber.java.en.Given; | import io.cucumber.java.en.Given; | ||||||
| import io.cucumber.java.en.Then; | import io.cucumber.java.en.Then; | ||||||
| import io.cucumber.java.en.When; | import io.cucumber.java.en.When; | ||||||
|  | import org.openqa.selenium.By; | ||||||
|  | import org.openqa.selenium.WebElement; | ||||||
|  | import org.openqa.selenium.support.ui.ExpectedConditions; | ||||||
|  | import org.openqa.selenium.support.ui.WebDriverWait; | ||||||
| 
 | 
 | ||||||
| public class FavoriteShopList { | public class FavoriteShopList { | ||||||
|     @Given("^the user has at least (\\d+) favorite shop$") |     @Given("^the user has at least (\\d+) favorite shop$") | ||||||
|     public void the_user_has_at_least_favorite_shop(int arg1) throws Exception { |     public void the_user_has_at_least_favorite_shop(int arg1) throws Exception { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @When("^the user clicks on favorite shops$") |     @Then("^the profile page should open$") | ||||||
|     public void the_user_clicks_on_favorite_shops() throws Exception { |     public void the_profile_page_should_open() throws Exception { | ||||||
|  |         WebElement profile_info_text = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("table.table.table-hover"))); | ||||||
|  |         assert(profile_info_text.isDisplayed()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^he should see his favorite shops list$") |     @Then("^he should see his favorite shops list$") | ||||||
|     public void he_Should_see_his_favorite_shops_list() throws Exception { |     public void he_should_see_his_favorite_shops_list() throws Exception { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @When("^he clicks on delete a favorite shop entry$") |     @When("^he clicks on delete a favorite shop entry$") | ||||||
|     public void he_clicks_on_delete_a_favorite_shop_enty() throws Exception { |     public void he_clicks_on_delete_a_favorite_shop_entry() throws Exception { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the favorite shop entry should be deleted$") |     @Then("^the favorite shop entry should be deleted$") | ||||||
|  |  | ||||||
|  | @ -4,4 +4,5 @@ import org.openqa.selenium.WebDriver; | ||||||
| 
 | 
 | ||||||
| public class Preconditions { | public class Preconditions { | ||||||
|     public static WebDriver driver; |     public static WebDriver driver; | ||||||
|  |     public static final int delaySeconds = 7; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -4,6 +4,12 @@ import io.cucumber.java.PendingException; | ||||||
| import io.cucumber.java.en.Given; | import io.cucumber.java.en.Given; | ||||||
| import io.cucumber.java.en.Then; | import io.cucumber.java.en.Then; | ||||||
| import io.cucumber.java.en.When; | import io.cucumber.java.en.When; | ||||||
|  | import org.openqa.selenium.By; | ||||||
|  | import org.openqa.selenium.WebElement; | ||||||
|  | import org.openqa.selenium.support.ui.ExpectedConditions; | ||||||
|  | import org.openqa.selenium.support.ui.WebDriverWait; | ||||||
|  | 
 | ||||||
|  | import java.util.List; | ||||||
| 
 | 
 | ||||||
| public class PriceAlarm { | public class PriceAlarm { | ||||||
|     @Given("^the user has at least (\\d+) price alarm set$") |     @Given("^the user has at least (\\d+) price alarm set$") | ||||||
|  | @ -12,42 +18,43 @@ public class PriceAlarm { | ||||||
| 
 | 
 | ||||||
|     @When("^the user clicks on the profile icon$") |     @When("^the user clicks on the profile icon$") | ||||||
|     public void the_user_clicks_on_the_profile_icon() throws Exception { |     public void the_user_clicks_on_the_profile_icon() throws Exception { | ||||||
|     } |         WebElement profileButton = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
| 
 |                 .until(ExpectedConditions.elementToBeClickable(By.xpath("//*[contains(text(),'profile')]"))); | ||||||
|     @Then("^the profile details popup should open$") |         profileButton.click(); | ||||||
|     public void the_profile_details_popup_should_open() throws Exception { |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @When("^the user clicks on price alarms$") |  | ||||||
|     public void the_user_clicks_on_price_alarms() throws Exception { |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Then("^the price alarm list should open$") |  | ||||||
|     public void the_price_alarm_list_should_open() throws Exception { |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the price alarm list should contain at least (\\d+) entry$") |     @Then("^the price alarm list should contain at least (\\d+) entry$") | ||||||
|     public void the_price_alarm_list_should_contain_at_least_entry(int arg1) throws Exception { |     public void the_price_alarm_list_should_contain_at_least_entry(int arg1) throws Exception { | ||||||
|  |         WebElement alarmEntry = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("table.table.table-hover tr:nth-child(2)"))); | ||||||
|  | 
 | ||||||
|  |         assert (alarmEntry != null); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the price alarm list should contain a maximum of (\\d+) entries per page$") |     @Given("^the user is on the profile page$") | ||||||
|     public void the_price_alarm_list_should_contain_a_maximum_of_entries_per_page(int arg1) throws Exception { |     public void the_user_is_on_the_profile_page() throws Exception { | ||||||
|     } |         Preconditions.driver.get("https://www.betterzon.xyz/profile"); | ||||||
| 
 | 
 | ||||||
|     @Given("^the user is on the price alarm list page$") |         WebElement profile_info_text = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|     public void the_user_is_on_the_price_alarm_list_page() throws Exception { |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("table.table.table-user-information"))); | ||||||
|  |         assert (profile_info_text.isDisplayed()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @When("^the user clicks on the \"([^\"]*)\" button next to a price alarm$") |     @When("^the user clicks on the \"([^\"]*)\" button next to a price alarm$") | ||||||
|     public void the_user_clicks_on_the_button_next_to_a_price_alarm(String arg1) throws Exception { |     public void the_user_clicks_on_the_button_next_to_a_price_alarm(String arg1) throws Exception { | ||||||
|  |         if (arg1.equals("remove")) { | ||||||
|  |             WebElement entry = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                     .until(ExpectedConditions.elementToBeClickable(By.cssSelector("table.table.table-hover tr:nth-child(2)"))); | ||||||
|  | 
 | ||||||
|  |             if (entry == null) { | ||||||
|  |                 throw new Exception("Too few price alarm entries found!"); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|     @Then("^a popup should open asking the user to confirm the removal$") |             WebElement btn = entry.findElement(By.cssSelector("img.delete[src='../assets/images/Delete_icon-icons.com_55931.png']")); | ||||||
|     public void a_popup_should_open_asking_the_user_to_confirm_the_removal() throws Exception { |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     @When("^the user confirms the removal of the price alarm$") |             btn.click(); | ||||||
|     public void the_user_confirms_the_removal_of_the_price_alarm() throws Exception { |         } else if (arg1.equals("edit")) { | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the price alarm should be removed from the database$") |     @Then("^the price alarm should be removed from the database$") | ||||||
|  |  | ||||||
|  | @ -4,9 +4,7 @@ import io.cucumber.java.PendingException; | ||||||
| import io.cucumber.java.en.Given; | import io.cucumber.java.en.Given; | ||||||
| import io.cucumber.java.en.Then; | import io.cucumber.java.en.Then; | ||||||
| import io.cucumber.java.en.When; | import io.cucumber.java.en.When; | ||||||
| import org.openqa.selenium.By; | import org.openqa.selenium.*; | ||||||
| import org.openqa.selenium.Keys; |  | ||||||
| import org.openqa.selenium.WebElement; |  | ||||||
| import org.openqa.selenium.support.ui.ExpectedConditions; | import org.openqa.selenium.support.ui.ExpectedConditions; | ||||||
| import org.openqa.selenium.support.ui.WebDriverWait; | import org.openqa.selenium.support.ui.WebDriverWait; | ||||||
| 
 | 
 | ||||||
|  | @ -15,8 +13,8 @@ public class SearchProduct { | ||||||
|     public void the_user_is_on_the_landing_page() throws Exception { |     public void the_user_is_on_the_landing_page() throws Exception { | ||||||
|         //throw new PendingException(); |         //throw new PendingException(); | ||||||
|         Preconditions.driver.get("https://betterzon.xyz"); |         Preconditions.driver.get("https://betterzon.xyz"); | ||||||
|         WebElement logo = (new WebDriverWait(Preconditions.driver, 10)) |         WebElement logo = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".navbar-brand"))); |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("a.navbar-brand"))); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @When("^the user enters the search term \"([^\"]*)\" and clicks search$") |     @When("^the user enters the search term \"([^\"]*)\" and clicks search$") | ||||||
|  | @ -24,49 +22,99 @@ public class SearchProduct { | ||||||
|         WebElement searchField = Preconditions.driver.findElement(By.cssSelector(".ng-untouched.ng-pristine.ng-valid")); |         WebElement searchField = Preconditions.driver.findElement(By.cssSelector(".ng-untouched.ng-pristine.ng-valid")); | ||||||
|         searchField.sendKeys(searchTerm); |         searchField.sendKeys(searchTerm); | ||||||
|         searchField.sendKeys(Keys.ENTER); |         searchField.sendKeys(Keys.ENTER); | ||||||
|         WebElement logo = (new WebDriverWait(Preconditions.driver, 10)) |         WebElement logo = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".navbar-brand"))); |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".navbar-brand"))); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the user should see the error page \"([^\"]*)\"$") |     @Then("^the user should see the error page \"([^\"]*)\"$") | ||||||
|     public void the_user_should_see_the_error_page(String arg0) throws Exception { |     public void the_user_should_see_the_error_page(String arg0) throws Exception { | ||||||
|         WebElement noProdsFoundMsg = (new WebDriverWait(Preconditions.driver, 10)) |         WebElement noProdsFoundMsg = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".ng-star-inserted"))); |                 .until(ExpectedConditions.elementToBeClickable(By.xpath("//*[contains(text(),'No Products found!')]"))); | ||||||
|         assert(noProdsFoundMsg.getText().contains("No Products found!")); |         assert (noProdsFoundMsg.isDisplayed()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Given("^the user is not logged in$") |     @Given("^the user is not logged in$") | ||||||
|     public void the_user_is_not_logged_in() throws Exception { |     public void the_user_is_not_logged_in() throws Exception { | ||||||
|  |         try { | ||||||
|  |             WebElement logoutButton = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                     .until(ExpectedConditions.elementToBeClickable(By.xpath("//*[contains(text(),'log out')]"))); | ||||||
|  | 
 | ||||||
|  |             logoutButton.click(); | ||||||
|  |         } catch (TimeoutException e) { | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Given("^the user is logged in$") |     @Given("^the user is logged in$") | ||||||
|     public void the_user_is_logged_in() throws Exception { |     public void the_user_is_logged_in() throws Exception { | ||||||
|  |         try { | ||||||
|  |             WebElement loginButton = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                     .until(ExpectedConditions.elementToBeClickable(By.xpath("//*[contains(text(),'sign in')]"))); | ||||||
|  |             loginButton.click(); | ||||||
|  | 
 | ||||||
|  |             WebElement usernameField = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                     .until(ExpectedConditions.elementToBeClickable(By.id("username"))); | ||||||
|  |             usernameField.sendKeys("Selenium"); | ||||||
|  | 
 | ||||||
|  |             WebElement passwordField = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                     .until(ExpectedConditions.elementToBeClickable(By.id("password"))); | ||||||
|  |             passwordField.sendKeys("Selenium"); | ||||||
|  | 
 | ||||||
|  |             WebElement loginBtn = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                     .until(ExpectedConditions.elementToBeClickable(By.className("btn_signin"))); | ||||||
|  |             loginBtn.click(); | ||||||
|  | 
 | ||||||
|  |         } catch (TimeoutException e) { | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the user should see a list of products$") |     @Then("^the user should see a list of products$") | ||||||
|     public void the_user_should_see_a_list_of_products() throws Exception { |     public void the_user_should_see_a_list_of_products() throws Exception { | ||||||
|         WebElement product = (new WebDriverWait(Preconditions.driver, 10)) |         WebElement product = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".productItem.ng-star-inserted"))); |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".row.p-2.bg-white.border.rounded"))); | ||||||
|         assert(product.isDisplayed()); |         assert (product.isDisplayed()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @When("^the user clicks on the first product$") |     @When("^the user clicks on the first product$") | ||||||
|     public void the_user_clicks_on_the_first_product() throws Exception { |     public void the_user_clicks_on_the_first_product() throws Exception { | ||||||
|  |         WebElement productDetailsBtn = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector(".row.p-2.bg-white.border.rounded button.btn.btn-primary.btn-sm"))); | ||||||
|  |         productDetailsBtn.click(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the user should see the product detail page$") |     @Then("^the user should see the product detail page$") | ||||||
|     public void the_user_should_see_the_product_detail_page() throws Exception { |     public void the_user_should_see_the_product_detail_page() throws Exception { | ||||||
|  |         WebElement productTitle = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("div.productTitle"))); | ||||||
|  |         assert (productTitle.isDisplayed()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the set price alarm box should show \"([^\"]*)\"$") |     @Then("^the set price alarm box should show \"([^\"]*)\"$") | ||||||
|     public void the_set_price_alarm_box_should_show(String arg0) throws Exception { |     public void the_set_price_alarm_box_should_show(String arg0) throws Exception { | ||||||
|  |         WebElement alarmBox = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("div.priceAlarm"))); | ||||||
|  |         if (arg0.equals("Login to set a price alarm")) { | ||||||
|  |             assert (alarmBox.getText().equals("Login to set a price alarm")); | ||||||
|  |         } else { | ||||||
|  |             assert (alarmBox.isDisplayed()); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @When("^the user sets a price alarm$") |     @When("^the user sets a price alarm$") | ||||||
|     public void the_user_sets_a_price_alarm() throws Exception { |     public void the_user_sets_a_price_alarm() throws Exception { | ||||||
|  |         WebElement alarmBoxField = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("div.priceAlarm input"))); | ||||||
|  |         alarmBoxField.sendKeys("12345"); | ||||||
|  |         WebElement alarmBox = (new WebDriverWait(Preconditions.driver, Preconditions.delaySeconds)) | ||||||
|  |                 .until(ExpectedConditions.elementToBeClickable(By.cssSelector("div.priceAlarm"))); | ||||||
|  |         alarmBox.click(); | ||||||
|  | 
 | ||||||
|  |         assert (alarmBox.isDisplayed() && alarmBoxField.isDisplayed()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Then("^the user should receive an email confirming the price alarm$") |     @Then("^the user should receive an email confirming the price alarm$") | ||||||
|     public void the_user_should_receive_an_email_confirming_the_price_alarm() throws Exception { |     public void the_user_should_receive_an_email_confirming_the_price_alarm() throws Exception { | ||||||
|  |         assert (true); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -5,8 +5,7 @@ Feature: Favorite Shop List | ||||||
|     And the user is logged in |     And the user is logged in | ||||||
|     And the user has at least 1 favorite shop |     And the user has at least 1 favorite shop | ||||||
|     When the user clicks on the profile icon |     When the user clicks on the profile icon | ||||||
|     Then the profile details popup should open |     Then the profile page should open | ||||||
|     When the user clicks on favorite shops |  | ||||||
|     Then he should see his favorite shops list |     Then he should see his favorite shops list | ||||||
| 
 | 
 | ||||||
|   Scenario: Remove Favorite Shop Entry |   Scenario: Remove Favorite Shop Entry | ||||||
|  | @ -14,7 +13,6 @@ Feature: Favorite Shop List | ||||||
|     And the user is logged in |     And the user is logged in | ||||||
|     And the user has at least 1 favorite shop |     And the user has at least 1 favorite shop | ||||||
|     When the user clicks on the profile icon |     When the user clicks on the profile icon | ||||||
|     Then the profile details popup should open |     Then the profile page should open | ||||||
|     When the user clicks on favorite shops |     When he clicks on delete a favorite shop entry | ||||||
|     And he clicks on delete a favorite shop entry |  | ||||||
|     Then the favorite shop entry should be deleted |     Then the favorite shop entry should be deleted | ||||||
|  |  | ||||||
|  | @ -5,23 +5,22 @@ Feature: Price Alarms | ||||||
|         And the user is logged in |         And the user is logged in | ||||||
|         And the user has at least 1 price alarm set |         And the user has at least 1 price alarm set | ||||||
|         When the user clicks on the profile icon |         When the user clicks on the profile icon | ||||||
|         Then the profile details popup should open |         Then the profile page should open | ||||||
|         When the user clicks on price alarms |  | ||||||
|         Then the price alarm list should open |  | ||||||
|         And the price alarm list should contain at least 1 entry |         And the price alarm list should contain at least 1 entry | ||||||
|         And the price alarm list should contain a maximum of 20 entries per page |  | ||||||
| 
 | 
 | ||||||
|     Scenario: Remove a price alarm |     Scenario: Remove a price alarm | ||||||
|         Given the user is on the price alarm list page |         Given the user is on the landing page | ||||||
|         And the user is logged in |         And the user is logged in | ||||||
|  |         When the user clicks on the profile icon | ||||||
|  |         Then the profile page should open | ||||||
|         When the user clicks on the "remove" button next to a price alarm |         When the user clicks on the "remove" button next to a price alarm | ||||||
|         Then a popup should open asking the user to confirm the removal |  | ||||||
|         When the user confirms the removal of the price alarm |  | ||||||
|         Then the price alarm should be removed from the database |         Then the price alarm should be removed from the database | ||||||
| 
 | 
 | ||||||
|     Scenario: Edit a price alarm |     Scenario: Edit a price alarm | ||||||
|         Given the user is on the price alarm list page |         Given the user is on the landing page | ||||||
|         And the user is logged in |         And the user is logged in | ||||||
|  |         When the user clicks on the profile icon | ||||||
|  |         Then the profile page should open | ||||||
|         When the user clicks on the "edit" button next to a price alarm |         When the user clicks on the "edit" button next to a price alarm | ||||||
|         Then a popup should open where the user can edit the alarm |         Then a popup should open where the user can edit the alarm | ||||||
|         When the user clicks on the "save changes" button |         When the user clicks on the "save changes" button | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ Feature: Search a Product | ||||||
|         Then the user should see a list of products |         Then the user should see a list of products | ||||||
|         When the user clicks on the first product |         When the user clicks on the first product | ||||||
|         Then the user should see the product detail page |         Then the user should see the product detail page | ||||||
|         And the set price alarm box should show "Log in to continue" |         And the set price alarm box should show "Login to set a price alarm" | ||||||
| 
 | 
 | ||||||
|     Scenario: User is logged in, searches for known product |     Scenario: User is logged in, searches for known product | ||||||
|         Given the user is on the landing page |         Given the user is on the landing page | ||||||
|  |  | ||||||
|  | @ -1,6 +1,6 @@ | ||||||
| # Betterzon | # Betterzon | ||||||
| Website: https://www.betterzon.xyz<br> | Website: https://betterzon.p4ddy.com<br> | ||||||
| Blog: https://blog.betterzon.xyz<br> | Blog: https://blog.betterzon.p4ddy.com<br> | ||||||
| Wiki: https://github.com/Mueller-Patrick/Betterzon/wiki | Wiki: https://github.com/Mueller-Patrick/Betterzon/wiki | ||||||
| 
 | 
 | ||||||
| # Code Quality | # Code Quality | ||||||
|  | @ -8,5 +8,5 @@ Wiki: https://github.com/Mueller-Patrick/Betterzon/wiki | ||||||
| [](https://ci.betterzon.xyz) | [](https://ci.betterzon.xyz) | ||||||
| 
 | 
 | ||||||
| # Project Status | # Project Status | ||||||
| [](https://www.betterzon.xyz) | [](https://betterzon.p4ddy.com) | ||||||
| [](https://blog.betterzon.xyz) | [](https://blog.betterzon.p4ddy.com) | ||||||
|  |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								important_files/Betterzon_Blog_PDF_Export.zip
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								important_files/Betterzon_Blog_PDF_Export.zip
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								important_files/Betterzon_FP_Calculation.xlsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								important_files/Betterzon_FP_Calculation.xlsx
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
		Loading…
	
		Reference in New Issue
	
	Block a user