[Android][iOS] Social pages editor fix #3248
No reviewers
Labels
No labels
Accessibility
Accessibility
Address
Address
Android
Android
Android Auto
Android Auto
Android Automotive (AAOS)
Android Automotive (AAOS)
API
API
AppGallery
AppGallery
AppStore
AppStore
Battery and Performance
Battery and Performance
Blocker
Blocker
Bookmarks and Tracks
Bookmarks and Tracks
Borders
Borders
Bug
Bug
Build
Build
CarPlay
CarPlay
Classificator
Classificator
Community
Community
Core
Core
CrashReports
CrashReports
Cycling
Cycling
Desktop
Desktop
DevEx
DevEx
DevOps
DevOps
dev_sandbox
dev_sandbox
Directions
Directions
Documentation
Documentation
Downloader
Downloader
Drape
Drape
Driving
Driving
Duplicate
Duplicate
Editor
Editor
Elevation
Elevation
Enhancement
Enhancement
Epic
Epic
External Map Datasets
External Map Datasets
F-Droid
F-Droid
Fonts
Fonts
Frequently User Reported
Frequently User Reported
Fund
Fund
Generator
Generator
Good first issue
Good first issue
Google Play
Google Play
GPS
GPS
GSoC
GSoC
iCloud
iCloud
Icons
Icons
iOS
iOS
Legal
Legal
Linux Desktop
Linux Desktop
Linux packaging
Linux packaging
Linux Phone
Linux Phone
Mac OS
Mac OS
Map Data
Map Data
Metro
Metro
Navigation
Navigation
Need Feedback
Need Feedback
Night Mode
Night Mode
NLnet 2024-06-281
NLnet 2024-06-281
No Feature Parity
No Feature Parity
Opening Hours
Opening Hours
Outdoors
Outdoors
POI Info
POI Info
Privacy
Privacy
Public Transport
Public Transport
Raw Idea
Raw Idea
Refactoring
Refactoring
Regional
Regional
Regression
Regression
Releases
Releases
RoboTest
RoboTest
Route Planning
Route Planning
Routing
Routing
Ruler
Ruler
Search
Search
Security
Security
Styles
Styles
Tests
Tests
Track Recording
Track Recording
Translations
Translations
TTS
TTS
UI
UI
UX
UX
Walk Navigation
Walk Navigation
Watches
Watches
Web
Web
Wikipedia
Wikipedia
Windows
Windows
Won't fix
Won't fix
World Map
World Map
No milestone
No project
No assignees
2 participants
Due date
No due date set.
Dependencies
No dependencies set.
Reference: organicmaps/organicmaps-tmp#3248
Loading…
Add table
Reference in a new issue
No description provided.
Delete branch "issue-2787-android-social-pages-editor"
Deleting a branch is permanent. Although the deleted branch may continue to exist for a short time before it actually gets removed, it CANNOT be undone in most cases. Continue?
Closes #2787, closes #3089
The problem is some URLs after normalization are not valid because domain name is removed. I fixed the issue by adding domain name to native code called from UI. Also domain name is added to OSM changeset now.
Facebook:
https://facebook.com/pages/Daikichi-Japans-Restaurant/118163444928138
pages/Daikichi-Japans-Restaurant/118163444928138
https://facebook.com/pages/Daikichi-Japans-Restaurant/118163444928138
pages/Daikichi-Japans-Restaurant/118163444928138
pages/Daikichi-Japans-Restaurant/118163444928138
https://facebook.com/pages/Daikichi-Japans-Restaurant/118163444928138
Instagram:
https://www.instagram.com/p/BvkgKZNDbqN/?ghid=UwPchX7B
p/BvkgKZNDbqN
https://instagram.com/p/BvkgKZNDbqN
p/BvkgKZNDbqN
p/BvkgKZNDbqN
https://instagram.com/p/BvkgKZNDbqN
Facebook 2:
www.facebook.com/OpenStreetMap
OpenStreetMap
@OpenStreetMap
OpenStreetMap
OpenStreetMap
Also fixed iPhone UI. This is my first commit to Swift and Objective-C code. Please verify!!!
Doesn't require map data change (in contrast to PR #3212).
Update 2023-01-23
Update 2023-01-31
@
symbol nowconstexpr
variables in C++ codeThis function converts
string_view value
tostring
even if no changes are made. Don't know how to improve it.Actually, I have a better idea. To avoid duplicating the code in both iOS and Android, let's do normalization in the core before features are uploaded to OSM. WDYT?
Also, are there some other checks which we can do in the core once instead of on both platforms?
@biodranik let me answer you comment here.
Here're analytics of
contact:facebook
data for the whole Planet:141 882
- total object in OSM database;5 949 920
- Total length ofcontact:facebook
values;2 498 146
- Total length of normalized values;8 302
- number of normalized values with '/' symbol (e.g.pg/LiveGalleryHairDesign
); it's 5.8% of all objects;2 672 488
- Total length of normalized values withhttps://facebook.com
domain (if it's needed); nicknames are not affected.So if we add
https://facebook.com/
explicitly to all URLs (not nicknames) it will increase map size by2 672 488 - 2 498 146 = 174 342
bytes.For all social contacts:
contact:facebook
141 882
5 949 920
2 498 146
8302
(5.8%)2 672 488
contact:instagram
69 168
2 590 348
895 059
255
(0.3%)900 669
contact:twitter
34 020
996 019
353 032
142
(0.4%)355 872
contact:vk
53 760
1 361 593
568 377
15
(0.03%)568 602
N.B.: I used Google BigQuery to get numbers. Here are SQLs osm_bigquery_facebook_stats.sql, osm_bigquery_instagram_stats.sql, osm_bigquery_twitter_stats.sql, osm_bigquery_vk_stats.sql.
Thanks for the stats. There is no need to increase entropy by copypasting thousands of lines of code into the OSM DB and our maps. When tens of millions of users start using OM, the number of these strings will grow quickly. Also, don't forget about the general rule: the more compact map data, the faster the rendering.
Can we solve that issue even easier? For example:
Here we see that we can strip anything, including @ symbol, and then restore the full link by adding https://www.facebook,com/ without breaking it (the link will remain valid).
https://facebook.com/
label before the input, so the user should type only the normalized part of the URL. But if someone inserts a full URL - no problem! Our code will send it to OSM as is, either a short form of the link, or a full one.What do you think about this approach?
@biodranik your description looks rational. It mostly works in such way on master branch.
95% of cases social contact field has only username or pagename. OM works fine with such values
@username
is shown on PlacePage ✅But there are 5% of cases when social contact field has no username but full URL. We strip domain name leaving short part
https://www.facebook.com/pages/category/Restaurant/Au-poincaré-182381015649241
->pages/category/Restaurant/Au-poincaré-182381015649241
With this PR I'm fixing cases 3. and 4.
And I'm thinking that short URL to full URL conversion could be moved to Native code to simplify UI code.
@biodranik
But from code perspective it's not hard to keep short pagename and full URL in parallel. See my code for a simple check.
In this PR social contacts work in such way:
I need to look iOS code if it needs modifications. Will install xCode and learn Swift.
Thanks for your efforts! I'm worrying that our code complexity grows with all these copy-pasted approaches, and it increases the time to support it. Can it be optimized?
What about Line?
Can we optimize and implement these similar checks only once, in the C++?
Why do you need it?
Or even better
return value;
if it compiles.ditto
switch is more appropriate here.
It would be great to avoid hardcoding these URL parts several times everywhere in the code.
It looks like a lot of duplicating code in the UI part. Can it be improved by querying the properly formatted string from the C++ core? E.g.
GetShortSocialLink(metadataId)
?@ -240,2 +243,3 @@
- (void)openFacebook:(PlacePageData *)data {
[self.ownerViewController openUrl:[NSString stringWithFormat:@"https://m.facebook.com/%@", data.infoData.facebook]];
std::string const fullUrl = osm::socialContactToURL(osm::MapObject::MetadataID::FMD_CONTACT_FACEBOOK, [data.infoData.facebook UTF8String]);
[self.ownerViewController openUrl:ToNSString(fullUrl)];
Can this code be optimized, so we implement the cross-platform logic only once, instead of duplicating it on each platform every time when we add some new social network support? Something like that:
Android or iOS parts have a generic method OpenLink, that internally calls some C++
Feature::FullSocialUrl(metadataId)
returning an URL in a string (and throwing an exception/crashing if an invalid metadataId was queried), and then opening the browser for it.Removed
const
.Done!
Removed parentheses
Replaced with
switch
Done
return {value};
doesn't work. Fails withchosen constructor is explicit in copy-initialization
This test verifies that OSM XML contains full URL
https://facebook.com/pages/Stereo-Plaza/118100041593935
instead ofpages/Stereo-Plaza/118100041593935
. Current implementation uploads latter value.@biodranik sorry for being offline with this PR. So this is another version of my fix:
socialContactToURL(...)
function to convert any social contant to full URL. Seevalidate_and_format_contacts.cpp
for details. All string constants with domain names are hardcoded there.socialContactToURL(...)
:2.1. On Place page if social contact is not pagename/username but a link
2.2. On Place page when user taps social contant and we build URL to open in browser/app
2.3. On Place page when user long tap on social contact
2.4. On Editor UI if social contact is not pagename/username but a link
@
symbol if social contact is an URLN.B. Every call to
socialContactToURL(...)
isn't cheap because of string convertions and native call:jstring -> std::string -> socialContactToURL(...) -> std::string -> jstring
or
NSString -> std::string -> socialContactToURL(...) -> std::string -> NSString
But UI calls this method only once per social network and only on user interaction: open Place Page view, tap, long tap, open Editor.
Is there a way to get a properly formatted URL for UI without the overhead? E.g. to query it by id, and receive the final, properly formatted one?
Is there any way I can read selected
mapObject
from native code? For editor there'sg_editableMapObject
variable in native code but not for PlacePage.I rebased this PR to the latest 'master'. And fixed case of
contact:facebook=facebook.com/profile.php?id=34496388
.Thank you very much for your work! I know that it already took a lot of time and energy for everyone, so there is even bigger motivation to finish it in the best way possible :)
Let's focus on how we can simplify everything.
Our shortened values can contain slashes without URLs, right? Like
page/something
.Do we really need to convert social values into full URLs for iOS and Android? Can this conversion be avoided completely? For example:
What do you think?
Is a mobile URL really required? Aren't desktop URLs automatically redirected to their mobile versions when opened? Removing this method simplifies the implementation.
This method is not needed if we don't show the full URL in the UI, right? Nobody needs to see a long full URL...
ditto
All these lines can be replaced by
Will storing the full URL in the map object simplify the code? Or storing it in the Editor is better? If we need full URL only when a link is clicked, then it should be enough, no?
Showing full URL in UI doesn't look great.
The UI code on all platforms will be greatly simplified if UI just shows whatever is in the data, without any conditions and formatting.
Would it be easier to avoid any logic here, and store it as-is? But apply the stripping logic (in C++ core) when a user presses "Save" in the editor window?
@ -10,3 +10,4 @@
TEST_EQUAL(osm::ValidateAndFormat_facebook("facebook.com/OpenStreetMap"), "OpenStreetMap", ());
TEST_EQUAL(osm::ValidateAndFormat_facebook("www.facebook.com/OpenStreetMap"), "OpenStreetMap", ());
TEST_EQUAL(osm::ValidateAndFormat_facebook("www.facebook.fr/OpenStreetMap"), "OpenStreetMap", ());
TEST_EQUAL(osm::ValidateAndFormat_facebook("http://facebook.com/OpenStreetMap"), "OpenStreetMap", ());
This link is not valid anymore.
@ -61,3 +89,3 @@
{
if (!v.empty() && !IsProtocolSpecified(v))
return kWebsiteProtocols[kWebsiteProtocolDefaultIndex] + v;
return string{kHttp}.append(v);
A faster version of this code is possible, but if we don't use it at all, it would be even better (see my other comment).
Is it a normal case? Maybe
ASSERT(!value.empty(), ());
could be more helpful?string_view objects are faster to compare than raw C strings. It makes sense to use local constexpr constants like this:
constexpr std::string_view kInstagram{"contact:instagram"};
if (tag == kInstagram) ...
{} is a safer constructor version than () here and above.
As hostnames are copy-pasted here, they can be stored as global static constexpr constants and reused in different methods.
Isn't Line normalized? Why do we store URLs for it?
Actually, it can be safe to assume that all URLs are normalized in the new release (after adding force-normalization into the generator) because now we don't allow users to edit the map if a new maps data update is available.
Do we really need a mobile url version?
nit: would the
switch
be better/faster?Can we avoid using full urls here?
@ -423,3 +428,4 @@
text:value
placeholder:name
keyboardType:UIKeyboardTypeDefault
capitalization:UITextAutocapitalizationTypeSentences];
Is this conversion really necessary?
Will removing @ simplify the code?
@ -250,2 +254,4 @@
std::string const fullUrl = osm::socialContactToURL(osm::MapObject::MetadataID::FMD_CONTACT_TWITTER, [data.infoData.twitter UTF8String]);
[self.ownerViewController openUrl:ToNSString(fullUrl)];
}
As I mentioned before, this code can be rewritten something like this:
(Need to think about where the actual method will be and how it will be called).
Currently selected Place Page can always be accessed from C++ code like this:
It may make sense to implement full URL conversion into the place page Info object, if it is not needed anywhere else.
If I understand you correctly, you propose to get rid of social network domain on PlacePage UI. This is how it will look (Android). I added options when URL has no "https://" and with "instagram" instead of "instagram.com". I also added one more option with leading
'/'
symbol.Which of those options do you prefer?
Without a slash is ok. It is not critical at all, because a better solution will be to show icons without the text. It will be obvious, and won't take much space, thus easier to use.
We can finish with this PR, create an issue about redesigning icons in Place Page, and do it later separately.
Fixed formatting.
Ok, I did as we agreed. Place page shows only part of URL. See screenshots in PR description.
Removed
else
.Fixed formatting.
Removed mobile URL generation all over the code. Only full URL is used.
Fixed formatting.
Simplified
PlacePageView.refreshSocialLinks(...)
method. Do not show full URL on PlacePage.Done!
Added method
MapObject.getContactUrl(Metadata.MetadataType)
to simplify the code.Storing short link is better. We show it on UI and can restore full URL any time (on click or in Editor).
Simplified with
MapObject.getContactUrl(Metadata.MetadataType)
callYes. PlacePage now shows social contacts with no modification. Only
@
symbol is added if needed.We do strip full URL in Editor. We can't send to OSM partial URLs like
contact:instagram=p/BvkgKZNDbqN
. Wiki says that either username of full URL is expected. This part of code converts all shortened URLs to full URLs before sending to OSM API.@ -10,3 +10,4 @@
TEST_EQUAL(osm::ValidateAndFormat_facebook("facebook.com/OpenStreetMap"), "OpenStreetMap", ());
TEST_EQUAL(osm::ValidateAndFormat_facebook("www.facebook.com/OpenStreetMap"), "OpenStreetMap", ());
TEST_EQUAL(osm::ValidateAndFormat_facebook("www.facebook.fr/OpenStreetMap"), "OpenStreetMap", ());
TEST_EQUAL(osm::ValidateAndFormat_facebook("http://facebook.com/OpenStreetMap"), "OpenStreetMap", ());
Fixed ID in the link.
@ -61,3 +89,3 @@
{
if (!v.empty() && !IsProtocolSpecified(v))
return kWebsiteProtocols[kWebsiteProtocolDefaultIndex] + v;
return string{kHttp}.append(v);
Fixed.
This code is used from
editor/xml_feature.cpp
to convert short URLs to full URLs for OSM API. I simplified it withconstexpr
as you proposed.You mean replace:
with
? I don't know how it would affect performance. We can try and use
switch
.Added
constexpr
for all"contact:*"
strings invalidate_and_format_contacts.cpp
.Fixed.
Fixed.
Line normalization is tricky. You can find some examples in
validate_and_format_contacts_test.cpp
file. E.g.https://line.me/R/ti/p/gdltt7s380
→ can extract usernamegdltt7s380
https://line.me/R/home/public/profile?id=r90ck7n1rq
→ can extract usernamer90ck7n1rq
https://liff.line.me/1645278921-kWRPP32q/?accountId=673watcr
→ no username and different domainhttps://page.line.me/?accountId=673watcr
→ different domain andaccountId
is not the same thing as username.So we store domain (no HTTP(S) protocol) for
contact:line
.Removed mobile URL code
Fixed formatting.
Reverted this code. Copy raw data here.
@ -423,3 +428,4 @@
text:value
placeholder:name
keyboardType:UIKeyboardTypeDefault
capitalization:UITextAutocapitalizationTypeSentences];
As we do with Android, short URLs should be converted to full URLs in editor. This is what issue #2787 is about.
Editor validates those values and stores short URLs.
Yes, removing
@
symbol would make code more simple. But for consistency with Android I'd rather keep it.I simplified this code by introducing
func prependAtIfNeed(_ contact: String) -> String
Fixed
@ -61,3 +89,3 @@
{
if (!v.empty() && !IsProtocolSpecified(v))
return kWebsiteProtocols[kWebsiteProtocolDefaultIndex] + v;
return string{kHttp}.append(v);
Fixed
I added
ASSERT()
and added @NonNull to the argument ofEditor.nativeGetFullUrl(int id, @NonNull String value)
I don't think that Platform is needed below. The previous include was correct.
I know that you want to fix a warning, but it's not a trivial one. We likely need to rename the Android header with a different name. But let's leave it for a separate PR.
Are you sure here? Let's leave it for another PR.
Do we really need to pass value from the Java here? Why
g_editableMapObject.GetMetadata(metaID);
can't be used to get it from the C++ code?Is it a valid scenario? If not, let's log some warning here.
Actually, the previous code did not check for null. It would be better to crash here and fix the bug, instead of debugging strange behavior later, right?
res can be extracted in nativeGetFullUrl, right?
As we discussed before, appending @ may confuse users. Don't use any @ symbols can make the code and UI cleaner. We can leave it for another PR.
Can
Editor.nativeGetFullUrl(Metadata.MetadataType.FMD_CONTACT_FACEBOOK);
be used here directly? What are the benefits of mMapObject proxy?Do you find @ symbol useful? It takes important horizontal space )
Does it make sense to add tests for other social networks too? Especially for strange Line cases?
Moving "profile.php" to constexpr string_view may make some comparisons faster.
Let's avoid the copy-paste by using constexpr string_view for hosts.
That's not important. It's nit )
What's the best way to concat two
string_view
values? Stackoverflow proposes convertion tostring
.gives error:
invalid operands to binary expression ('const std::string_view' (aka 'const basic_string_view<char>') and 'std::string_view' (aka 'basic_string_view<char>'))
string{urlInstagram} + value
should help. You can't join views, they just point to some other data.Introduced
constexpr
variables for "profile.php", domain names, URLs etc.Removed
mMapObject.getContactUrl()
method. IntroducedFramework.nativeGetPoiContactUrl(id)
to simplify code.Introduced
Framework.nativeGetPoiContactUrl()
function insteadI saw compiler warnings and changed include statement.
Reverting it back.
Ok, reverting
Removed
@
symbol from Android and iOS place page UIWe have tests for normalization code:
instagram.com/some_caffe_uk
->some_caffe_uk
(for all social networks)And I added tests for denormalization code:
some_caffe_uk
->https://instagram.com/some_caffe_uk
(for all social networks)This test verifies that normalization was applied (reading from OSM XML) and then denormalization (writing to OSM XML). I think we don't need to add all cases here. Just to make sure that input data is normalized and output is denormalized back.
N.B. Output values not always equal to input when we talk about OSM XML.
Almost done!
@Jean-BaptisteC can you please test this PR?
Is there a unit-test for this method?
According to our docs/CPP_STYLE.md:
So all constants should be named like kConstant.
It is not critical, but it is better to avoid unnecessary memory allocation for a temporary string object. The optimal way to do it (here and in other places):
Replaced
+
with.append(...)
in the file.Ok, didn't see style guide.
Renamed constants in the file.
Added tests for EndsWith(string, string_view)
You've made it! 🎉
nit: if the variable is not modified, and is not optimized by RVO (returned from the function), then we always mark it as const.
string_view const
orconst string_view
? I'm confused with C++ const semantics 😊It is the same :) But we use the right-to-left notation (check our CPP_STYLE.md for details).
For example:
string_view const * const
reads as const pointer to a const string_view.Done!
Pixel 6 - Android 13 ✅
Facebook:
Tested with https://www.openstreetmap.org/node/2864998473
Instagram:
Tested with https://www.openstreetmap.org/node/2864998473
Twitter:
https://www.openstreetmap.org/node/9722726898
VK:
https://www.openstreetmap.org/node/3199685169
LINE:
https://www.openstreetmap.org/node/4007245809
Long press on Place page works (Copy just ID and copy full URL) ✅
We can't add instagram link in Facebook field (Tested on all socials fields) ✅
I have create account on master dev osm and add in OM debug, but changes are not upload
I suspect that updating kothic submodule is not needed in this PR.
Don't commit it, just make git submodule update ...
UPD: And twine is also not needed here.
@ -332,0 +333,4 @@
return EndsWith(s1, std::string_view(s2));
}
bool EndsWith(std::string const & s1, std::string_view s2)
Now you can simply call EndsWith(s1, std::string_view(s2)) here
Here is the shorter way to construct string from view:
std::string contactFacebook(view);
{ from new line
Looks like this PR updates submodules kothic and twine. It’s by accident probably after rebase. Need to revert submodule changes…
Update: Done!
@ -332,0 +333,4 @@
return EndsWith(s1, std::string_view(s2));
}
bool EndsWith(std::string const & s1, std::string_view s2)
Fixed
Fixed
Fixed formatting
@ -332,2 +335,4 @@
bool EndsWith(std::string const & s1, std::string_view s2)
{
size_t const n = s1.size();
Replace all the function's body here :)
@ -332,2 +335,4 @@
bool EndsWith(std::string const & s1, std::string_view s2)
{
size_t const n = s1.size();
Replaced body 😊