-
Notifications
You must be signed in to change notification settings - Fork 124
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
Fix: [Capacitor] Set overscroll background color from theme #2594
base: main
Are you sure you want to change the base?
Conversation
@raineorshine The difference in puppeteer tests on this pr is caused by turning off the bounce feature of ios and taking inspiration from @ankitkarna99's proposed solution in the issue: Issue 1700. Originally, it is suggested to set scroll to false, but this caused issues with the webView. Instead, I opted for a solution similar to this, disabling the scroll bounce within the ios webview and setting the scrolling power to within our app itself. We cannot set the background color dynamically on the ios webview as we do with javascript when a user changes themes, so we have to opt for a solution like this which either hides or limits the user from seeing the webview background behind the app. With these changes, there is a drawback. By disabling the scroll bounce, it disables the scroll refresh feature which naturally comes with the Ios Webview. @raineorshine please tell me your thoughts on this issue based on this information. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the update! This looks like a promising approach. The overscroll background color now correctly matches the theme background color based on my testing.
We cannot set the background color dynamically on the ios webview as we do with javascript when a user changes themes, so we have to opt for a solution like this which either hides or limits the user from seeing the webview background behind the app.
Got it. That makes sense, and good to know the limitations with the WebView.
With these changes, there is a drawback. By disabling the scroll bounce, it disables the scroll refresh feature which naturally comes with the Ios Webview.
That's fine, we don't use that feature.
The only issue I am seeing is that programmatic scroll functionality is broken. Here are some examples, although there are likely others if window.scrollTo
is not working everywhere.
- Scroll zone parallax - When the user scrolls, the scroll zone on the right should scroll independently, at a reduced rate.
- scrollCursorIntoView - When the cursor moves to a thought that is outside the viewport, it should automatically scroll to the thought. To reproduce, activate New Subthought on a thought with a long list of children that exceeds the height of the viewport. It should scroll down to the new thought.
- etc
It looks like it's working. Simulator.Screen.Recording.-.iPhone.15.Pro.-.2024-11-18.at.13.10.11.mp4 |
Hmmm sorry about that! Maybe my branch was not up-to-date, or I failed to build the Capacitor app correctly. I do see now that the parallax scrolling and scrollCursorIntoVIew are working as expected. Now what I'm seeing in the XCode simulator is that scroll bounce on overscroll is disabled completely. The scroll stops when it reaches the top. Just to confirm, is that what you have as well? |
@raineorshine Are you talking about where you are scroll to the very bottom and create a new sub-thought? Like this: Simulator.Screen.Recording.-.iPhone.15.Pro.-.2024-11-18.at.13.52.54.mp4I also thought this was not expected, but when I double-checked if the same behavior happens on main I found that it's exactly the same as my branch. |
I'm referring to the "scroll bounce" or "elastic scroll" behavior that occurs when scrolling past the top of the document. Here is the behavior on trim.2994A651-A7B9-4032-900E-6FF8C26DC760.MOV |
@raineorshine yes, that scroll bounce is actually a part of the UIScrollView of the IOS device. It is disabled in IOS in order to prevent the overscroll background from being shown since we can't dynamically alter the color of that (Its not javascript.) This is only an issue with IOS devices so this fix is here. Here is a video of the scroll bounce enabled and the UIScrollView background color set to white for visibility: Simulator.Screen.Recording.-.iPhone.15.Pro.-.2024-11-18.at.14.10.38.mp4 |
I see, thanks. Unfortunately we can't disable the scroll bounce. I've tried it both ways, and it feels really unnatural without it. I'm open to other ideas, though I realize we are somewhat limited given the behavior of the WebView! Maybe it's possible to add scroll bounce to a container within the body. |
@raineorshine I think that's possible. However, I'm not as familiar with adding animation and transformations with React as I am with Angular. I don't think I can complete this ticket since we can't remove the scroll bounce. Do you want to reopen a new ticket to add a scroll bounce to the native JavaScript? |
When you say animations and transformations, do you mean the native scroll bounce or a Javascript emulation? Because the Javascript emulations I've seen are never quite performant enough to mimic native functionality.
Are you wanting me to remove it from the milestone?
Let's keep the original issue so the full history is there. Changing the way the scroll bounce works is not so much a separate issue as a part of fixing the overscroll background behavior without breaking the existing functionality. |
@raineorshine I mean the javascript emulation. It looks needlessly complicated to implement and might interact weirdly with the UIScrollView of IOS. The problem with adding the javascript bounce is that it will affect all previous app versions on safari, web, android, and Ios. Where as right now, this is just an IOS issue. In fact, by disabling the scroll bounce on IOS it is made consistent with all the other versions of the app. There is no scroll bounce similar on web right now and this would be adding a new change across the app. |
I've looked at all the Javascript emulations that exist for scroll bounce and unfortunately none of them perform well enough to replace the native scroll bounce. We're going to have to stick to native browser scroll bounce. Since this appears to not be possible on the WebView without losing control of the overscroll background, then it should be on an element in the DOM. I think we should explore this latter possibility.
Not true. We currently have scroll bounce in Chrome, Safari, and Mobile Safari PWA, which is how I use the app personally on a daily basis. em has always had scroll bounce, at least on Apple devices. |
@raineorshine Alright, I managed to fix this issue after a lot of pain. I created a small custom plugin for capacitor and attached it to the project to interact with the native Webview. |
For some reason that mp4 is not working for me. We do set the background color right after the app loads, which happens right after the webview loads. In capacitor when the app launches it first loads -> Webview(Native code) -> App (React) |
How about this? What I'm seeing is that the overscroll background color is black when the theme is Light after the app loads and after the splash screen is gone, as the user is using the app. black-overscroll-bg-simulator.mov |
Wow, that's really strange. I'm glad you caught that. That shouldn't be happening. When I pulled the latest code and pushed up a fix for it. For some reason, the setting of the theme was not happening and I needed read it back in in the AppComponent. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you, @msdewitt!
The overscroll background color appears to be fixed, and the scroll bounce functions as expected 🎉
Both scroll zone parallax
and scrollCursorIntoView
are also working as expected 👍
ios.ScreenRecording_12-31-2024.00-23-48_1.mp4
1. Initial black overscroll is not resolved until first interaction
The black overscroll is initially displayed when reopening the app in Light mode, and it does not rerender to the correct color until the user interacts with the app. I understand from earlier comments that a limitation with Capacitor's configuration prevents us from resolving the overscroll color until the app/JS loads, but it shouldn't need to wait for user interaction.
Demo:
overscroll.ScreenRecording_12-31-2024.01-07-53_1.mp4
2. Thoughts displayed above toolbar
When scrolling down, thoughts appear above the scrollbar. They should be out of view once they scroll into/above the toolbar.
Current / bug-issue-1700
:
Expected / main
:
3. Bulky node_modules
diff is still in the commit history
Thousands of files were introduced in 973195d and subsequently removed in a5941fd. There may be other commits with this issue, but these are the two I found.
As originally mentioned in this comment, please rebase (squashing 973195d and a5941fd together is one option) to avoid bloating the repository's history with these changes.
## Install | ||
|
||
```bash | ||
npm install webview-background |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would prefer if we used yarn
(possibly other tools/libraries?) consistently with the main project, but I am fine with this if @raineorshine doesn't mind.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, it would be preferable to use the same package manager throughout the project.
Originally suggested here: #2594 (comment)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@msdewitt -- We still have the package-lock.json
/ npm configuration in this package. Can we please switch to yarn?
}); | ||
export * from './definitions'; | ||
export { WebviewBackground }; | ||
//# sourceMappingURL=index.js.map |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would strongly prefer not to include dist
files in the repository, and to include building this package as part of building the main project. There are a variety of reasons (repository bloat, merge conflicts more likely, built code more difficult to review for security issues, difficulty verifying that the build matches the source, etc.) for this, but they're relatively unimportant for this particular package due to its small size. I'll defer to @raineorshine who previously indicated that this is probably fine 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That was my intuition as well, so I'm glad you said something. I know that @msdewitt did this to simplify the webview
package release process, but I would love to discuss the feasibility of building it from source within the root project's build pipeline.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Awesome
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@msdewitt -- Would it be feasible to build from source (as part of the main project's yarn build
) rather than committing the dist
build to the repo? I imagine we could update the main package.json
build
command to something like:
"build": "yarn build:webview && yarn build:styles && vite build",
Then add the build:webview
script in there as well to build this package from source.
"unpkg": "dist/plugin.js", | ||
"files": [ | ||
"android/src/main/", | ||
"android/build.gradle", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is there an android
directory? I don't see one.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Its not needed for this plugin. We can add it later if needed. I deleted it since the fix seemed to only be for IOS.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Makes sense. We can remove the android
files from this list in package.json
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still seeing these which can be removed:
"android/src/main/",
"android/build.gradle",
Alright, I put in the fixes you requested @trevinhofmann . I had to go and change my approach to hide the contentInset on IOS with an ios only CSS adjustments. I hope to get the approved soon after these changes. I'll rebase after the changes are confirmed in this rework. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you, @msdewitt!
These issues appear to be resolved:
- Initial black overscroll is not resolved until first interaction
- Thoughts displayed above toolbar
The node_modules
commits will still need to be rebased/squashed for (3).
4. Toolbar has been shifted down
The toolbar at the top of the page has been shifted down considerably and needs to be positioned back to its original location.
toolbar.mp4
@@ -29,8 +29,6 @@ const PuppeteerEnvironment: Environment = { | |||
.catch((err: Error) => { | |||
// using `console.log` here to avoid errors or logs being swallowed by vitest | |||
// all of `console.error`, `console.warn` and `console.info` don't show up in the terminal | |||
// eslint-disable-next-line no-console | |||
console.log('Could not connect to browserless.') |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this change relevant to the PR?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, for some reason lint didn't like it after some changes. Most likely after I merged with main.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We probably want to keep this logging in, as it informs the user if there is any issue with connecting to browserless. The linter does not complain in main
, so I suspect that this is a regression.
@@ -50,8 +50,6 @@ const setup = async ({ | |||
break | |||
case 'info': | |||
case 'log': | |||
// eslint-disable-next-line no-console | |||
console[messageType](text) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this change relevant to the PR?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same as above. Lint didn't like it.
import { WebPlugin } from '@capacitor/core'; | ||
export class WebviewBackgroundWeb extends WebPlugin { | ||
async changeBackgroundColor(options) { | ||
console.log('Color changed: ', options.color); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like this log was removed as requested from the source (packages/webview/src/web.ts
) but then not rebuilt and included in the build here.
One example of why we should build from source rather than including the dist
in the repo :)
this.animateAndClose!() | ||
this.props.onClose?.() | ||
if (Capacitor.isNativePlatform() && fullReload) { | ||
window.location.reload() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't believe we should be doing a full page reload here. The only cases where we have those at the moment are for unrecoverable error handling.
Is there a particular reason for doing this? If I understand correctly, it's doing a full reload every time the Settings modal is closed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We have already had discussions about this in this pr. Its unfortunate that we have to do this, but it is necessary. We can only limit the changes down to the very minimal as possible. (I.E. Only in the settings menu with a flag set as true.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is due to a limitation in changing the background color through the WebView. It's bad, but it's the only way to update the overscroll background color from what I understand.
It should reload the page only when the theme setting has changed.
@@ -221,9 +222,11 @@ const Toolbar: FC<ToolbarProps> = ({ customize, onSelect, selected }) => { | |||
}), | |||
)} | |||
style={{ | |||
// height: fontSize + 300, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Unintentional comment? This can be removed.
@trevinhofmann Please tell me the IOS version you are using to test that toolbar being moved down. In order to cope with your previous review changes about not showing the notes when scrolling down, I had to add a feature which alters the CSS to make the navbar look like it is showing so that it is not covered by the native IOS statusBar. For some reason, the device you are using is showing that change, but the native IOS StatusBar already has a cover, therefor showing it as too big. |
I am using an iPhone 14 Pro with iOS version 18.1.1. |
Alright, maybe its an issue for anything below Ios 15 I need to check. |
@trevinhofmann Could you try and run the simulator with IOS version 17? I think the issue happens with versions later than 18.0, but I need to confirm that. I tried to replicate it locally with an Iphone 14 but it didn't show the same screen error as your screen capture showed. |
I can confirm the toolbar is shifted down on iOS 18.1 and iOS 17.5 in XCode Simulator. |
@raineorshine How about IOS 17.4? I am using 17.4 on Iphone 15. It might also have todo with Iphone Pro versions of the simulator. That might be something I overlooked. |
Each Simulator runtime is ~6-8 GB so I'm not able to download additional minor versions to test. However, I am able to reproduce the issue on the following versions:
|
@msdewitt -- Thank you again for the PR. The toolbar reposition appears to be reproducible on several iOS versions. Please let me know if you need any additional assistance. |
Issue 1700
I added a way to eliminate the ios bounce outside the boundary of the app while adding a scroll feature within the controllable range of the app code instead of the device.