Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add Saudi Arabia national shields #851

Draft
wants to merge 10 commits into
base: main
Choose a base branch
from
56 changes: 56 additions & 0 deletions icons/shield_sa_national.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
4 changes: 3 additions & 1 deletion src/js/screen_gfx.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,9 @@

import rgba from "color-rgba";

const fontFamily = '"sans-serif-condensed", "Arial Narrow", sans-serif';
const naskhFonts =
'"Noto Naskh Arabic", KacstNaskh, "Sakkal Majalla", "DecoType Naskh Regular"';
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I picked a few Naskh fonts that are available by default on most operating systems. I'm not 100% sure these are all the best choices, but they look close enough to the style as shown on Saudi shields.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Legibility doesn't look good on my system:

image

Why not use the packaged Americana fontstacks that have Arabic?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shields are rendered client-side using fonts available to the browser. We would have to re-engineer shields to render text from a PBF fontstack instead of standard web formats.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh duh good point. Don't you need some kind of <link> tag to use the Google Noto fonts?

Copy link
Member

@ZeLonewolf ZeLonewolf Apr 8, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Something like

<link href='https://fonts.googleapis.com/css?family=Noto Naskh Arabic' rel='stylesheet'>

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not if it's already installed on the system, as with Android. But downloading and displaying Noto on all systems is an option.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there a reason we wouldn't? I assume it's just a one time download into browser cache.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I suppose not. Though we might as well include Noto Sans Armenian as well, if consistency is what we're going for.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I guess we'll need whatever scripts would get rendered into shields.

const fontFamily = `"sans-serif-condensed", "Arial Narrow", ${naskhFonts}, sans-serif`;
export const shieldFont = (size) => `bold ${size}px ${fontFamily}`;
export const fontSizeThreshold = 12;

Expand Down
28 changes: 26 additions & 2 deletions src/js/shield.js
Original file line number Diff line number Diff line change
Expand Up @@ -303,6 +303,23 @@ export function getRouteDef(id) {
};
}

/**
* Reformats an alphanumeric ref as Eastern Arabic numerals, preserving any
* alphabetic suffix.
*/
export const latinToArabicDigits = (ref) =>
`${ref}`
.replaceAll("0", "\u0660")
.replaceAll("1", "\u0661")
.replaceAll("2", "\u0662")
.replaceAll("3", "\u0663")
.replaceAll("4", "\u0664")
.replaceAll("5", "\u0665")
.replaceAll("6", "\u0666")
.replaceAll("7", "\u0667")
.replaceAll("8", "\u0668")
.replaceAll("9", "\u0669");
claysmalley marked this conversation as resolved.
Show resolved Hide resolved

/**
* Reformats an alphanumeric ref as Roman numerals, preserving any alphabetic
* suffix.
Expand Down Expand Up @@ -382,8 +399,15 @@ export function generateShieldCtx(map, id) {
// Convert numbering systems. Normally alternative numbering systems should be
// tagged directly in ref=*, but some shields use different numbering systems
// for aesthetic reasons only.
if (routeDef.ref && shieldDef.numberingSystem === "roman") {
routeDef.ref = romanizeRef(routeDef.ref);
if (routeDef.ref) {
switch (shieldDef.numberingSystem) {
case "arab":
routeDef.ref = latinToArabicDigits(routeDef.ref);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Roads in Saudi Arabia are signed solely in Eastern Arabic numerals. This PR contains code to convert digits to Eastern Arabic.

In #676, I included a Roman numeral conversion function because the Roman numerals were purely decorative. In plain text (and even some signs), the route numbers are in Western Arabic numerals. By contrast, Eastern Arabic is the local language’s numeral system. Would it be more appropriate to render the number however it’s tagged in ref, allowing mappers the flexibility to shunt the transliterated number to a subkey such as ref:Latn?

/ref https://github.com/ZeLonewolf/openstreetmap-americana/issues/467#issuecomment-1169234746

Copy link
Member

@ZeLonewolf ZeLonewolf Apr 9, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a philosophical question indeed. Off the cuff I would say the ref should have the Eastern Arabic numbers in them, which we would render with the right font. This PR takes a more resilient approach and papers over the difference at the expense of not exposing what we might agree is an error.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It would be worth asking the folks that implemented the scheme though I'm not sure who to ask.

Copy link
Member

@1ec5 1ec5 Apr 9, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note that most of Saudi Arabia’s highways are also part of the Arab Mashreq International Road Network, which is signposted and tagged in Western Arabic numerals: #466.

break;
claysmalley marked this conversation as resolved.
Show resolved Hide resolved
case "roman":
routeDef.ref = romanizeRef(routeDef.ref);
break;
}
}

// Add the halo around modifier plaque text
Expand Down
14 changes: 14 additions & 0 deletions src/js/shield_defs.js
Original file line number Diff line number Diff line change
Expand Up @@ -3687,6 +3687,20 @@ export function loadShields() {
},
};

// Saudi Arabia
shields["SA:national"] = {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Currently, only one national route is mapped with network=SA:national, and all other routes are missing either the network=* tag or a relation altogether. But SA:national clearly represents a coherent network, and this could encourage local mappers to complete the tags for other routes.

The one network=SA:national route is what Saudi Arabia calls a “highway” route, as opposed to a “city road” or “branch road”. All three classifications are part of a nationwide system. Part of Highway 40 has been mapped redundantly as network=SA:highway. If we could get a local mapper to confirm their preference, that might also have the effect of encouraging more build-out of these route relations.

spriteBlank: "shield_sa_national",
textLayout: textConstraint("diamond"),
textColor: Color.shields.black,
numberingSystem: "arab",
padding: {
left: 3,
right: 2,
top: 4,
bottom: 3,
},
};

// Turkey
shields["TR:motorway"] = hexagonVerticalShield(
2,
Expand Down
4 changes: 4 additions & 0 deletions src/shieldtest.js
Original file line number Diff line number Diff line change
Expand Up @@ -232,6 +232,10 @@ let networks = [
"AU:QLD:ALT",
"AU:QLD:ALT_S",
"AU:VIC:ALT_NR",

// With other numeral systems
"IN:NE",
"SA:national",
];

// Uncomment for a list of all supported networks. This makes for a very long page.
Expand Down