Jump to content

Recommended Posts

Posted (edited)

Greetings, folks!

Here is a way to optimise the HTML length by up to 30%, depending on the nesting.

The examples I give are taken from an old Interlude core, but the general idea is applicable to all java-based cores.

Proof of concept!
The two important bits in the screenshot below are the (1) Original HTML size row and the (4) No Tabs/Rows size row, which represent the before and after, respectively.

image.png.79a2d793c47dfd3fabd786d0b6cd2399.png

STEPS:

 1. locate the `setHtml(String text)` method inside `NpcHtmlMessage.java`.

 2. add the following bit of code as a standalone method:

/**
 * Replaces all occurrences of New Rows and Tab spaces in the string.
 * @param text the string that may contain invalid characters
 * @return the string with invalid character replaced by underscores
 */
public static String removeTabSpacesAndNewRows(String text)
{
	return text.replaceAll("[\\t\\n]", "");
}


 3. find the last ocurence of the `text` String variable in the `setHtml()` method, and push it through this newly created method like so: `removeTabSpacesAndNewRows(text);`.
 
4. example of the final result.
 

/**
 * Sets the html.
 * @param text the new html
 */
public void setHtml(String text)
{
	if (text == null)
	{
		LOGGER.warning("Html is null! this will crash the client!");
		_html = "<html><body></body></html>";
		return;
	}
	
	if (text.length() > 8192)
	{
		LOGGER.warning("Html is too long! this will crash the client!");
		_html = "<html><body>Html was too long,<br>Try to use DB for this action</body></html>";
		return;
	}
		
	_html = removeTabSpacesAndNewRows(text); // html code must not exceed 8192 bytes
}

/**
 * Replaces all occurrences of New Rows and Tab spaces in the string.
 * @param text the string that may contain invalid characters
 * @return the string with invalid character replaced by underscores
 */
public static String removeTabSpacesAndNewRows(String text)
{
	return text.replaceAll("[\\t\\n]", "");
}



5. You could further refine it by processing the `text` variable between the two IF clauses by creating a new local variable, assigning it a value of `text` and then replacing the subsequent  `text` mentions in the `setHtml()` method with this new local var. Here's what it would look like:
 

/**
 * Sets the html.
 * @param text the new html
 */
public void setHtml(String text)
{
	if (text == null)
	{
		LOGGER.warning("Html is null! this will crash the client!");
		_html = "<html><body></body></html>";
		return;
	}

	String refinedText = removeTabSpacesAndNewRows(text);
	if (refinedText.length() > 8192)
	{
		LOGGER.warning("Html is too long! this will crash the client!");
		_html = "<html><body>Html was too long,<br>Try to use DB for this action</body></html>";
		return;
	}
		
	_html = refinedText; // html code must not exceed 8192 bytes
}



UPDATE:
(thanks to @xdem for pointing it out)

You can instead apply the same logic/method to the HTMCache.java
To do so, locate the loadFile() method and recycle the String/Text/Content through the removeTabSpacesAndNewRows method.

KEEP IN MIND that you might want to move the removeTabSpacesAndNewRows method in another java class, preferrably in some UTILITY class.
Here is an example:
 

	public String loadFile(File file)
	{
		final HtmFilter filter = new HtmFilter();
		String content = null;
		if (file.exists() && filter.accept(file) && !file.isDirectory())
		{
			FileInputStream fis = null;
			BufferedInputStream bis = null;
			try
			{
				fis = new FileInputStream(file);
				bis = new BufferedInputStream(fis);
				final int bytes = bis.available();
				final byte[] raw = new byte[bytes];
				bis.read(raw);
				
				content = new String(raw, StandardCharsets.UTF_8);
				content = content.replaceAll("\r\n", "\n");
				content = content.replaceAll("(?s)<!--.*?-->", ""); // Remove html comments
				content = NpcHtmlMessage.removeTabSpacesAndNewRows(content);
				
				final String relpath = Util.getRelativePath(Config.DATAPACK_ROOT, file);
				final int hashcode = relpath.hashCode();
				if (Config.CHECK_HTML_ENCODING && !StandardCharsets.US_ASCII.newEncoder().canEncode(content))
				{
					LOGGER.warning("HTML encoding check: File " + relpath + " contains non ASCII content.");
				}
				
				final String oldContent = _cache.get(hashcode);
				if (oldContent == null)
				{
					_bytesBuffLen += bytes;
					_loadedFiles++;
				}
				else
				{
					_bytesBuffLen = (_bytesBuffLen - oldContent.length()) + bytes;
				}
				
				_cache.put(hashcode, content);
			}
			catch (Exception e)
			{
				LOGGER.warning("Problem with htm file " + e);
			}
			finally
			{
				if (bis != null)
				{
					try
					{
						bis.close();
					}
					catch (Exception e1)
					{
						LOGGER.warning("Problem with HtmCache: " + e1.getMessage());
					}
				}
				
				if (fis != null)
				{
					try
					{
						fis.close();
					}
					catch (Exception e1)
					{
						LOGGER.warning("Problem with HtmCache: " + e1.getMessage());
					}
				}
			}
		}
		
		return content;
	}


 

Edited by Bru7aLMike
headline
  • Like 1
  • Thanks 1
Posted

okay its simple and it will work, but you better save the minified html directly in the HTML Cache during the load phase!

Posted
9 minutes ago, xdem said:

okay its simple and it will work, but you better save the minified html directly in the HTML Cache during the load phase!


Ok, I will update the guide then.

Posted

I would also advice you to experiment with

 

replaceAll("\\s+", " ");

 

s+ will automatically match any whitespace character (including \r \n etc)
 

Posted
25 minutes ago, xdem said:

I would also advice you to experiment with

 

replaceAll("\\s+", " ");

 

s+ will automatically match any whitespace character (including \r \n etc)
 



I have, as evident from the posted screenshot, see a copy below.

The bottom-most line is No White-Spaces, at all.
However, the 5% gain is not worth the risk, in my personal opinion. The Tab-Spaces and New Rows reduces the overall length by quite a lot as is.

image.png.748e32247416b582de08a1491e88a5da.png

Here is a list of all four methods that I used in the screenshot.

	public static String removeTabSpaces(String text)
	{
		return text.replaceAll("\\t", "");
	}

	public static String removeNewRows(String text)
	{
		return text.replaceAll("\\n", "");
	}

	public static String removeTabSpacesAndNewRows(String text)
	{
		return text.replaceAll("[\\t\\n]", "");
	}

	public static String removeAllWhiteSpacing(String text)
	{
		return text.replaceAll("\\s+", "");
	}

 

Posted

That's so useful in interlude clients since the html length is very low and you can't have a full menu page.

For example, this is in interlude but html code after editing to remove spaces and etc is unreadable.
It will make editing way much easier.
lootboxes-armors-nocturn-protojah.png

Thank you for your share 😇

  • Salty Mike changed the title to How to: reduce HTML weight/length from the core/java
  • 4 weeks later...

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now


  • Posts

    • ## [1.4.0] - 2026-01-28   ### ✨ New Features - **Vote System**: Lineage 2 servers can now use our vote–reward system. Players vote on the website and claim rewards in-game (1 vote = 1 claim) - **Vote Page**: On each server’s page (`/servers/<server>`), a **“Vote for Server”** button opens a dedicated vote page with cooldown info and optional Turnstile verification - **By Votes View**: The **“By Votes”** tab on the main page shows **actual vote counts** per server - **API Documentation**: New **API Docs** page at `/docs` (and footer link) with HMAC auth, endpoints, and examples for game server integration - **Vote API (My Servers)**: Server owners can open **“Vote API”** in My Servers to manage credentials, cooldown, allowed IPs, and open the docs   ### 🔄 Improvements - **Server Pages**: Single-server data is cached and loads faster; server pages can be opened by ID or by name (e.g. `/servers/my-server-name`) - **API Root**: Visiting the API root redirects to the docs URL configured in admin (default: site docs page) - **Admin Panel**: New **“Vote System”** tab for global settings (Turnstile, API security, default cooldown, docs URL)   ### 🔐 Security & Reliability - Turnstile (CAPTCHA) support for vote submissions to reduce abuse - HMAC-protected game server API for secure vote check/claim and stats
    • "I recently purchased the account panel from this developer and wanted to leave a positive review.   The transaction was smooth, and the developer demonstrated exceptional professionalism throughout the process.   What truly sets them apart is their outstanding post-sale support. They are responsive, patient, and genuinely helpful when addressing questions or issues. It's clear they care about their customers' experience beyond just the initial sale.   I am thoroughly satisfied and grateful for the service. This is a trustworthy seller who provides real value through both a quality product and reliable support. 100% recommended."
    • Server owners, Top.MaxCheaters.com is now live and accepting Lineage 2 server listings. There is no voting, no rankings manipulation, and no paid advantages. Visibility is clean and equal, and early listings naturally appear at the top while the platform grows. If your server is active, it should already be listed. Submit here https://Top.MaxCheaters.com This platform is part of the MaxCheaters.com network and is being built as a long-term reference point for the Lineage 2 community. — MaxCheaters.com Team
    • ⚙️ General Changed “No Carrier” title to “Disconnected” to avoid confusion after abnormal DC. On-screen Clan War kill notifications will no longer appear during Sieges, Epics, or Events. Bladedancer or SwordSinger classes can now log in even when Max Clients (2) is reached, you cannot have both at the same time. The max is 3 clients. Duels will now be aborted if a monster aggros players during a duel (retail-like behavior). Players can no longer send party requests to blocked players (retail-like). Fixed Researcher Euclie NPC dialogue HTML error. Changed Clan leave/kick penalty from 12 hours to 3 hours. 🧙 Skills Adjusted Decrease Atk. Spd. & Decrease Speed land rates in Varka & FoG. Fixed augmented weapons not getting cooldown when entering Olympiad. 🎉 Events New Team vs Team map added. New Save the King map added (old TvT map). Mounts disabled during Events. Letter Collector Event enabled Monsters drop letters until Feb. 13th Louie the Cat in Giran until Feb. 16th Inventory slots +10 during event period 📜 Quests Fixed “Possessor of a Precious Soul Part 1” rare stuck issue when exceeding max quest items. Fixed Seven Signs applying Strife buff/debuff every Monday until restart. 🏆 Milestones New milestone: “Defeat 700 Monsters in Varka” 🎁 Rewards: 200 Varka’s Mane + Daily Coin 🌍 NEW EXP Bonus Zones Hot Springs added Varka Silenos added (hidden spots excluded) As always, thank you for your support! L2Elixir keeps evolving, improving, and growing every day 💙   Website: https://l2elixir.org/ Discord: https://discord.gg/5ydPHvhbxs
  • Topics

×
×
  • Create New...

Important Information

This community uses essential cookies to function properly. Non-essential cookies and third-party services are used only with your consent. Read our Privacy Policy and We have placed cookies on your device to help make this website better. You can adjust your cookie settings, otherwise we'll assume you're okay to continue..