Compare commits
	
		
			No commits in common. "main" and "1.0.2" have entirely different histories.
		
	
	
		|  | @ -1,60 +0,0 @@ | ||||||
| name: Bug Report |  | ||||||
| description: Report a bug in the app. |  | ||||||
| labels: ["bug"] |  | ||||||
| body: |  | ||||||
|   - type: checkboxes |  | ||||||
|     attributes: |  | ||||||
|       label: "Requirements" |  | ||||||
|       description: "Please check all the following requirements before submitting a bug report." |  | ||||||
|       options: |  | ||||||
|         - label: I have searched the issues of this repository and believe that this is not a duplicate |  | ||||||
|           required: true |  | ||||||
|         - label: I have confirmed this bug exists on the latest version of the app |  | ||||||
|           required: true |  | ||||||
|   - id: platforms |  | ||||||
|     type: dropdown |  | ||||||
|     attributes: |  | ||||||
|       label: "Platforms" |  | ||||||
|       description: "The platforms on which the bug occurs or had occurred. Select all that apply." |  | ||||||
|       multiple: true |  | ||||||
|       options: |  | ||||||
|         - Android |  | ||||||
|         - Windows |  | ||||||
|         - Linux |  | ||||||
|         - Web |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - id: description |  | ||||||
|     type: input |  | ||||||
|     attributes: |  | ||||||
|       label: "Description" |  | ||||||
|       description: "A short description of what the bug is." |  | ||||||
|       placeholder: "I was trying to do X, but Y happened instead." |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Steps to reproduce" |  | ||||||
|       description: "Detailed steps for reproducing the issue." |  | ||||||
|       placeholder: "1. Go to '...' 2. Click on '....' 3. Scroll down to '....' 4. See error" |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Expected behavior" |  | ||||||
|       description: "A clear and concise description of what you expected to happen." |  | ||||||
|       placeholder: "I expected to see..." |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Actual behavior" |  | ||||||
|       description: "A clear and concise description of what actually happened." |  | ||||||
|       placeholder: "Instead, I saw..." |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - id: context |  | ||||||
|     type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Screenshots or additional context" |  | ||||||
|       description: "If applicable, add screenshots to help explain your problem." |  | ||||||
|  | @ -1,8 +0,0 @@ | ||||||
| blank_issues_enabled: false |  | ||||||
| contact_links: |  | ||||||
|   - name: Contact Author |  | ||||||
|     url: https://jhubi1.com/mailto |  | ||||||
|     about: Report security concerns or ask general questions. |  | ||||||
|   - name: Translations |  | ||||||
|     url: https://crowdin.com/project/ollama-app |  | ||||||
|     about: Help translate the app into your language. |  | ||||||
|  | @ -1,31 +0,0 @@ | ||||||
| name: Feature Request |  | ||||||
| description: Suggest a new feature or improvement for the app. |  | ||||||
| labels: ["enhancement"] |  | ||||||
| body: |  | ||||||
|   - type: checkboxes |  | ||||||
|     attributes: |  | ||||||
|       label: "Requirements" |  | ||||||
|       description: "Please check all the following requirements before submitting a feature request." |  | ||||||
|       options: |  | ||||||
|         - label: I have searched the issues of this repository and believe that this is not a duplicate |  | ||||||
|           required: true |  | ||||||
|         - label: I can confirm this feature request is not already implemented in the app |  | ||||||
|           required: true |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Description" |  | ||||||
|       description: "A description of the feature you are proposing." |  | ||||||
|       placeholder: "I would like to see..." |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Possible solution" |  | ||||||
|       description: "Suggestions on how to implement the feature." |  | ||||||
|       placeholder: "It would be great if..." |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: "Additional context" |  | ||||||
|       description: "Add any other context, screenshots or possible concepts about the feature request here." |  | ||||||
|  | @ -1,155 +0,0 @@ | ||||||
| name: Build app |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|     inputs: |  | ||||||
|       obfuscate: |  | ||||||
|         description: Obfuscate |  | ||||||
|         default: false |  | ||||||
|         type: boolean |  | ||||||
|       buildAndroid: |  | ||||||
|         description: Build for Android |  | ||||||
|         required: true |  | ||||||
|         default: true |  | ||||||
|         type: boolean |  | ||||||
|       buildWindowsX64: |  | ||||||
|         description: Build for Windows x64 |  | ||||||
|         required: true |  | ||||||
|         default: false |  | ||||||
|         type: boolean |  | ||||||
|       buildLinuxX64: |  | ||||||
|         description: Build for Linux x64 |  | ||||||
|         required: true |  | ||||||
|         default: false |  | ||||||
|         type: boolean |  | ||||||
| 
 |  | ||||||
| jobs: |  | ||||||
|   analyze: |  | ||||||
|     name: Linting |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     outputs: |  | ||||||
|       result: ${{ steps.version.outputs.result }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: subosito/flutter-action@v2.18.0 |  | ||||||
|         with: |  | ||||||
|           channel: stable |  | ||||||
|           flutter-version-file: pubspec.yaml |  | ||||||
|           cache: true |  | ||||||
|       - name: Get flutter version |  | ||||||
|         id: version |  | ||||||
|         uses: mikefarah/yq@master |  | ||||||
|         with: |  | ||||||
|           cmd: yq -r '.version' 'pubspec.yaml' | awk -F'+' '{print $1}' |  | ||||||
|       - name: Disabling flutter analytics |  | ||||||
|         run: flutter config --no-analytics |  | ||||||
|       - name: Analyzing project code |  | ||||||
|         run: flutter analyze --no-fatal-infos |  | ||||||
|   build-android: |  | ||||||
|     name: Building for Android |  | ||||||
|     if: ${{ github.event.inputs.buildAndroid == 'true' }} |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     needs: analyze |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: subosito/flutter-action@v2.18.0 |  | ||||||
|         with: |  | ||||||
|           channel: stable |  | ||||||
|           flutter-version-file: pubspec.yaml |  | ||||||
|           cache: true |  | ||||||
|       - uses: actions/setup-java@v4 |  | ||||||
|         with: |  | ||||||
|           distribution: "adopt" |  | ||||||
|           java-version: "17" |  | ||||||
|       - name: Copy keystore file |  | ||||||
|         run: | |  | ||||||
|           echo $'storePassword=${{ secrets.ANDROID_KEYSTORE_PASSPHRASE }}\nkeyPassword=${{ secrets.ANDROID_KEYSTORE_PASSPHRASE }}\nkeyAlias=upload\nstoreFile=upload-keystore.jks\n' > ./android/key.properties |  | ||||||
|           echo "${{ secrets.ANDROID_KEYSTORE }}" > ./android/app/upload-keystore.jks.asc |  | ||||||
|           gpg -d --passphrase "${{ secrets.ANDROID_KEYSTORE_PASSPHRASE }}" --batch ./android/app/upload-keystore.jks.asc > ./android/app/upload-keystore.jks |  | ||||||
|       - name: Disabling flutter analytics |  | ||||||
|         run: flutter config --no-analytics |  | ||||||
|       - name: Running build |  | ||||||
|         id: compile |  | ||||||
|         run: flutter build apk --split-debug-info=build/debugAndroid ${{ github.event.inputs.obfuscate == 'true' && '--obfuscate' || '' }} |  | ||||||
|       - name: Preparing files |  | ||||||
|         run: | |  | ||||||
|           cp build/app/outputs/flutter-apk/app-release.apk build/app/outputs/flutter-apk/ollama-android-v${{ needs.analyze.outputs.result }}.apk |  | ||||||
|       - name: Uploading APK |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: ollama-android |  | ||||||
|           path: | |  | ||||||
|             build/app/outputs/flutter-apk/ollama-android-v${{ needs.analyze.outputs.result }}.apk |  | ||||||
|   build-windows-x64: |  | ||||||
|     name: Building for Windows x64 |  | ||||||
|     if: ${{ github.event.inputs.buildWindowsX64 == 'true' }} |  | ||||||
|     runs-on: windows-latest |  | ||||||
|     needs: analyze |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: subosito/flutter-action@v2.18.0 |  | ||||||
|         with: |  | ||||||
|           channel: stable |  | ||||||
|           flutter-version-file: pubspec.yaml |  | ||||||
|           cache: true |  | ||||||
|       - name: Disabling flutter analytics |  | ||||||
|         run: flutter config --no-analytics |  | ||||||
|       - name: Running build |  | ||||||
|         id: compile |  | ||||||
|         run: flutter build windows --split-debug-info=build\debugWindows ${{ github.event.inputs.obfuscate == 'true' && '--obfuscate' || '' }} |  | ||||||
|       - name: Running installer build |  | ||||||
|         uses: Minionguyjpro/Inno-Setup-Action@v1.2.2 |  | ||||||
|         with: |  | ||||||
|           path: .\windows_installer\ollama.iss |  | ||||||
|           options: /O+ /dAppVersion=${{ needs.analyze.outputs.result }} |  | ||||||
|       - name: Uploading installer |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: ollama-windows-x64 |  | ||||||
|           path: build\windows\x64\runner\ollama-windows-x64-v${{ needs.analyze.outputs.result }}.exe |  | ||||||
|   build-linux-x64: |  | ||||||
|     name: Building for Linux x64 |  | ||||||
|     if: ${{ github.event.inputs.buildLinuxX64 == 'true' }} |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     needs: analyze |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: subosito/flutter-action@v2.18.0 |  | ||||||
|         with: |  | ||||||
|           channel: stable |  | ||||||
|           flutter-version-file: pubspec.yaml |  | ||||||
|           cache: true |  | ||||||
|       - name: Disabling flutter analytics |  | ||||||
|         run: flutter config --no-analytics |  | ||||||
|       - name: Installing linux dependencies |  | ||||||
|         run: | |  | ||||||
|           sudo apt-get update -y |  | ||||||
|           sudo apt-get install -y ninja-build libgtk-3-dev |  | ||||||
|       - name: Running build |  | ||||||
|         id: compile |  | ||||||
|         run: flutter build linux --split-debug-info=build/debugLinux ${{ github.event.inputs.obfuscate == 'true' && '--obfuscate' || '' }} |  | ||||||
|       - name: Creating archive |  | ||||||
|         run: | |  | ||||||
|           cd build/linux/x64/release/bundle |  | ||||||
|           tar -czf ollama-linux-x64-v${{ needs.analyze.outputs.result }}.tar.gz * |  | ||||||
|       - name: Uploading archive |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: ollama-linux-x64 |  | ||||||
|           path: build/linux/x64/release/bundle/ollama-linux-x64-v${{ needs.analyze.outputs.result }}.tar.gz |  | ||||||
|   bundle: |  | ||||||
|     name: Creating bundle |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     if: ${{ always() }} |  | ||||||
|     needs: [build-android, build-windows-x64, build-linux-x64] |  | ||||||
|     steps: |  | ||||||
|       - name: Adding builds |  | ||||||
|         uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           merge-multiple: true |  | ||||||
|       - name: Generating timestamp |  | ||||||
|         run: echo "timestamp=$EPOCHSECONDS"$'\n\norigin=${{ github.repository }}\nhost=${{ github.server_url }}\nowner=${{ github.repository_owner }}\n\nworkflow=${{ github.workflow }}\nrun_id=${{ github.run_id }}\nrun_number=${{ github.run_number }}' > manifest.yaml |  | ||||||
|       - name: Bundling files |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: ollama |  | ||||||
|           path: ./ |  | ||||||
							
								
								
									
										13
									
								
								.metadata
								
								
								
								
							
							
						
						|  | @ -4,7 +4,7 @@ | ||||||
| # This file should be version controlled and should not be manually edited. | # This file should be version controlled and should not be manually edited. | ||||||
| 
 | 
 | ||||||
| version: | version: | ||||||
|   revision: "35c388afb57ef061d06a39b537336c87e0e3d1b1" |   revision: "a14f74ff3a1cbd521163c5f03d68113d50af93d3" | ||||||
|   channel: "stable" |   channel: "stable" | ||||||
| 
 | 
 | ||||||
| project_type: app | project_type: app | ||||||
|  | @ -13,11 +13,14 @@ project_type: app | ||||||
| migration: | migration: | ||||||
|   platforms: |   platforms: | ||||||
|     - platform: root |     - platform: root | ||||||
|       create_revision: 35c388afb57ef061d06a39b537336c87e0e3d1b1 |       create_revision: a14f74ff3a1cbd521163c5f03d68113d50af93d3 | ||||||
|       base_revision: 35c388afb57ef061d06a39b537336c87e0e3d1b1 |       base_revision: a14f74ff3a1cbd521163c5f03d68113d50af93d3 | ||||||
|  |     - platform: android | ||||||
|  |       create_revision: a14f74ff3a1cbd521163c5f03d68113d50af93d3 | ||||||
|  |       base_revision: a14f74ff3a1cbd521163c5f03d68113d50af93d3 | ||||||
|     - platform: windows |     - platform: windows | ||||||
|       create_revision: 35c388afb57ef061d06a39b537336c87e0e3d1b1 |       create_revision: a14f74ff3a1cbd521163c5f03d68113d50af93d3 | ||||||
|       base_revision: 35c388afb57ef061d06a39b537336c87e0e3d1b1 |       base_revision: a14f74ff3a1cbd521163c5f03d68113d50af93d3 | ||||||
| 
 | 
 | ||||||
|   # User provided section |   # User provided section | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| { |  | ||||||
|     "recommendations": [ |  | ||||||
|         "dart-code.dart-code", |  | ||||||
|         "dart-code.flutter", |  | ||||||
|         "github.vscode-github-actions" |  | ||||||
|     ] |  | ||||||
| } |  | ||||||
							
								
								
									
										133
									
								
								README.md
								
								
								
								
							
							
						
						|  | @ -2,74 +2,129 @@ | ||||||
| 
 | 
 | ||||||
| A modern and easy-to-use client for Ollama. Have the greatest experience while keeping everything private and in your local network. | A modern and easy-to-use client for Ollama. Have the greatest experience while keeping everything private and in your local network. | ||||||
| 
 | 
 | ||||||
| |  |  |  |  | | |  |  |  |  | | ||||||
| |-|-|-|-| | |-|-|-|-| | ||||||
| 
 | 
 | ||||||
| > [!IMPORTANT] | > Important: This app does not host a Ollama server on device, but rather connects to one and uses its api endpoint. | ||||||
| > This app does not host a Ollama server on device, but rather connects to one using its api endpoint. | > Don't know what Ollama is? Learn more at [ollama.com](https://ollama.com/). | ||||||
| > You don't know what Ollama is? Learn more at [ollama.com](https://ollama.com). |  | ||||||
| 
 | 
 | ||||||
| ## Getting Started | - [Ollama App](#ollama-app) | ||||||
|  |   - [Installation](#installation) | ||||||
|  |   - [Initial Setup](#initial-setup) | ||||||
|  |   - [Side Menu](#side-menu) | ||||||
|  |   - [Model Selector](#model-selector) | ||||||
|  |   - [Multimodal Model Input](#multimodal-model-input) | ||||||
|  |   - [Multilingual Interface](#multilingual-interface) | ||||||
|  |   - [Custom Builds](#custom-builds) | ||||||
|  |     - [Actually Building](#actually-building) | ||||||
| 
 | 
 | ||||||
| Ollama App has a pretty simple and intuitive interface to be as open as possible. Everything just works out of the box, you just have to follow the next steps. | ## Installation | ||||||
| 
 | 
 | ||||||
| ### Installation | You'll find the latest recommended version of the Ollama App under [the releases tab](https://github.com/JHubi1/ollama-app/releases). Download the APK and install it on your Android device. That's it, now proceed to [Initial Setup](#initial-setup). | ||||||
| 
 | 
 | ||||||
| You'll find the latest recommended version of the Ollama App under the [releases tab](https://github.com/JHubi1/ollama-app/releases). Download the correct executable onto your device and install it. If you want to install on a desktop platform, you might also have to follow the steps listed below, under [Ollama App for Desktop](#ollama-app-for-desktop). | ## Initial Setup | ||||||
| 
 | 
 | ||||||
| Alternatively, you can also download the app from any of the following stores: | After installing the app and opening it for the first time, you'll encounter this popup: | ||||||
| 
 | 
 | ||||||
| [<img src="assets/stores/IzzyOnDroid.png" width="215" />](https://apt.izzysoft.de/fdroid/index/apk/com.freakurl.apps.ollama/) |  | ||||||
| <!-- [<img src="assets/stores/FDroid.png" width="215" />](/com.freakurl.apps.ollama/) --> |  | ||||||
| 
 | 
 | ||||||
| That's it, you've successfully installed Ollama App! Now just proceed to [Initial Setup](https://github.com/JHubi1/ollama-app/wiki/Getting-Started#initial-setup) or alternatively [Setup](#setup) below. | Go through the welcome dialog one by one, you should read their content, but you don't have to. | ||||||
| 
 | 
 | ||||||
| ### Ollama App for Desktop | |  |  |  | | ||||||
|  | |-|-|-| | ||||||
| 
 | 
 | ||||||
| > [!WARNING] | After going through that, you'll get a small snack bar notifying you that you have to set the host. For that, open the sidebar (swipe from the left to right or click the icon in the top left corner) and click on settings. There you'll find all app-related settings, you should go through them, but for the initial setup, only the first one is important. | ||||||
| > This is still an experimental feature! Some functions may not work as intended. If you come across any errors, please create a new [issue report](https://github.com/JHubi1/ollama-app/issues/new/choose). |  | ||||||
| 
 | 
 | ||||||
| There are a few things you might have to keep in mind if you're planning to use the experimental desktop support. | In the bit host text field, you have to enter the base URL of your instance. The port is required, except for the port number matching the protocol (443 for HTTPS or 80 for HTTP). After that, click the save icon right next to the text field. | ||||||
| 
 | 
 | ||||||
| #### Windows | This address will be checked, so no worry about entering the wrong one. The disadvantage of this is that your server has to be running even if you don't want to chat with it at that moment. If you set the host once, and your server is offline, the requests will fail, but the host will stay saved if you don't change it yourself. Don't worry, just go into the side menu and click the settings button to change it. | ||||||
| 
 | 
 | ||||||
| The Windows version is provided in the form of an installer, you can find it attached on the [latest release](https://github.com/JHubi1/ollama-app/releases). It's not signed, you might have to dismiss the Windows Defender screen by pressing "View More" > "Run Anyway". | That's it, you can now just chat. Enter a message into the box at the bottom and click the send icon. | ||||||
| 
 | 
 | ||||||
| Windows app data is kept at: `C:\Users\[user]\AppData\Roaming\JHubi1\Ollama App` | ## Side Menu | ||||||
| 
 | 
 | ||||||
| #### Linux | The button on the top left opens the menu. In it, you have two options: `New Chat` and `Settings`. The first option creates a new chat, and the second one opens the settings screen where you can change how everything works. | ||||||
| 
 | 
 | ||||||
| The Linux version is provided in the form of a portable executable, you can find it attached on the [latest release](https://github.com/JHubi1/ollama-app/releases). To start it, execute `./ollama` in the extracted folder. | Below that are all the chats. To delete one, swipe it from left to right. To rename the chat tab and hold it until a popup dialog appears. In it, you can change the title or tab the sparkle icon to let AI find one for you. This is not affected by the "generate title" setting. | ||||||
| 
 | 
 | ||||||
| If a message like `error while loading shared libraries: libgtk-3.so.0: cannot open shared object file` appears when you start the app, run the following commands: | | |  | | ||||||
|  | |-|-| | ||||||
| 
 | 
 | ||||||
| - `sudo apt-get update` | > Note: The button on the top right corner deletes the chat. It has the same effect as swiping the chat in the sidebar. | ||||||
| - `sudo apt-get upgrade` |  | ||||||
| - `sudo apt-get install packagekit-gtk3-module` |  | ||||||
| 
 | 
 | ||||||
| Linux app data is kept at: `/home/[user]/.local/share/ollama` | ## Model Selector | ||||||
| 
 | 
 | ||||||
| ### Setup | You can access the model selector by tapping on the `<selector>` text in the top middle or the name of the currently selected model in the same spot. Then you'll get the following bottom sheet: | ||||||
| 
 | 
 | ||||||
| The most difficult part is setting up the host. To learn more visit the [wiki guide on how to do so](https://github.com/JHubi1/ollama-app/wiki/Getting-Started#setting-up-the-host). After setting up, you normally don't have to enter it again. |  | ||||||
| 
 | 
 | ||||||
| And you're done! Just start chatting with your local AI and have fun! | This will display all the models currently installed in your Ollama server instance. | ||||||
| 
 | 
 | ||||||
| > [!TIP] | Models with an image-like icon next to them allow multimodal input. The one shown in the image, `llava`, supports exactly that. | ||||||
| > The new Voice Mode is now avaliable as an experimental feature. Learn more about it in [the documentation](https://github.com/JHubi1/ollama-app/wiki/Components#voice). |  | ||||||
| 
 | 
 | ||||||
| ## Documentation | The models with a star next to them are recommended models. They have been selected by me (hehe) to be listed as that. Read more under [Custom Builds](#custom-builds). | ||||||
| 
 | 
 | ||||||
| The documentation for components, settings, functions, etc. has moved to the [Wiki Page](https://github.com/JHubi1/ollama-app/wiki) of this repository. The steps there will be updated with future versions. Still having questions? Feel free to open an issue. | The `Add` button does nothing at the moment, it just opens a snack bar listing steps on how to add a model to an instance. For safety reasons, I didn't add the ability to add a model directly via name in the app. | ||||||
| 
 | 
 | ||||||
| ## Translations and Contribution | ## Multimodal Model Input | ||||||
| 
 | 
 | ||||||
| You want to help me make this project even better? Great, help is always appresheated. | Ollama App supports multimodal models, models with support input via an image. | ||||||
| 
 | 
 | ||||||
| Ollama App is created using [Flutter](https://flutter.dev), a modern and robust frontend framework designed to make a single codebase run on multiple target platforms. The framework itself is based on the [Dart](https://dart.dev) programming language. | After selecting a supported model, as describes in [Model Selector](#model-selector), a new icon appears at the bottom left of the message bar; a camera icon. Clicking on it reveals the following bottom sheet: | ||||||
| 
 | 
 | ||||||
| Read more in the [Contribution Guide](https://github.com/JHubi1/ollama-app/wiki/Contributing). | |  |  | | ||||||
|  | |-|-| | ||||||
| 
 | 
 | ||||||
| ## Star History | Select one of them, take or select your photo and it'll get added to the chat. You can also add multiple. | ||||||
| 
 | 
 | ||||||
|  | Even though the images will appear in the chat already after sending, they won't be submitted to the AI until a new text message is sent. | ||||||
|  | 
 | ||||||
|  | ## Multilingual Interface | ||||||
|  | 
 | ||||||
|  | I integrated support for multiple languages into the Ollama App. Currently available are: | ||||||
|  | 
 | ||||||
|  | - English | ||||||
|  | - German | ||||||
|  | 
 | ||||||
|  | Your language isn't one of them? Reach out to me and I'll give you access to my Crowdin project. | ||||||
|  | 
 | ||||||
|  | ## Custom Builds | ||||||
|  | 
 | ||||||
|  | Now comes the interesting part. I built this app in a way you can easily create custom builds. Currently, there are these values that can be customized: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | // use host or not, if false dialog is shown | ||||||
|  | const useHost = false; | ||||||
|  | // host of ollama, must be accessible from the client, without trailing slash, will always be accepted as valid | ||||||
|  | const fixedHost = "http://example.com:11434"; | ||||||
|  | // use model or not, if false selector is shown | ||||||
|  | const useModel = false; | ||||||
|  | // model name as string, must be valid ollama model! | ||||||
|  | const fixedModel = "gemma"; | ||||||
|  | // recommended models, shown with as star in model selector | ||||||
|  | const recommendedModels = ["gemma", "llama3"]; | ||||||
|  | // allow opening of settings | ||||||
|  | const allowSettings = true; | ||||||
|  | // allow multiple chats | ||||||
|  | const allowMultipleChats = true; | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | They can be found at the top of `lib/main.dart`. `useHost` and `useModel` decide whether you want `fixedHost` and `fixedModel` to control anything. `fixedHost` and `fixedModel` decide about the value that has to be used. That can be practical in case you try to create an app specific to your instance. | ||||||
|  | 
 | ||||||
|  | `recommendedModels` is a list of models that will be listed as recommended in the [Model Selector](#model-selector). They are more like personal preferences. If empty, no model will be preferred. | ||||||
|  | 
 | ||||||
|  | `allowSettings` will disable the settings screen. But it will also disable the welcome dialog at first startup and the ability to rename chats. | ||||||
|  | 
 | ||||||
|  | `allowMultipleChats` simply removes the `New Chat` option in the [Side Menu](#side-menu). And will load up the only available chat on app startup. | ||||||
|  | 
 | ||||||
|  | ### Actually Building | ||||||
|  | 
 | ||||||
|  | But how do you create a custom build? | ||||||
|  | 
 | ||||||
|  | First, follow [the Flutter installation guide](https://docs.flutter.dev/get-started/install) by selecting Android as the first app type. Then follow [these steps](https://docs.flutter.dev/deployment/android#signing-the-app) till you have your custom `key.properties`. Place it into the `android` folder at the root of the project. | ||||||
|  | 
 | ||||||
|  | If you're running on Windows, just double-click on `scripts/build.bat` and wait till the process is done. Don't worry, there'll be a lot of Kotlin errors in the terminal. You can safely ignore them, the build will be fine. | ||||||
|  | 
 | ||||||
|  | If you're not running Windows, open the file `scripts/build.bat` in a text editor and copy the command starting with `flutter` after the `call` command in a terminal window. Again, don't worry about the Kotlin errors. | ||||||
|  | 
 | ||||||
|  | In both cases, you'll now find your APK in `build/app/outputs/apk/release/app-release.apk` (don't blame me for that, it's a flutter thing). | ||||||
|  | @ -5,10 +5,9 @@ gradle-wrapper.jar | ||||||
| /gradlew.bat | /gradlew.bat | ||||||
| /local.properties | /local.properties | ||||||
| GeneratedPluginRegistrant.java | GeneratedPluginRegistrant.java | ||||||
| .cxx/ |  | ||||||
| 
 | 
 | ||||||
| # Remember to never publicly share your keystore. | # Remember to never publicly share your keystore. | ||||||
| # See https://flutter.dev/to/reference-keystore | # See https://flutter.dev/docs/deployment/android#reference-the-keystore-from-the-app | ||||||
| key.properties | key.properties | ||||||
| **/*.keystore | **/*.keystore | ||||||
| **/*.jks | **/*.jks | ||||||
|  |  | ||||||
|  | @ -0,0 +1,85 @@ | ||||||
|  | plugins { | ||||||
|  |     id "com.android.application" | ||||||
|  |     id "kotlin-android" | ||||||
|  |     id "dev.flutter.flutter-gradle-plugin" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | def localProperties = new Properties() | ||||||
|  | def localPropertiesFile = rootProject.file('local.properties') | ||||||
|  | if (localPropertiesFile.exists()) { | ||||||
|  |     localPropertiesFile.withReader('UTF-8') { reader -> | ||||||
|  |         localProperties.load(reader) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | def flutterVersionCode = localProperties.getProperty('flutter.versionCode') | ||||||
|  | if (flutterVersionCode == null) { | ||||||
|  |     flutterVersionCode = '1' | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | def flutterVersionName = localProperties.getProperty('flutter.versionName') | ||||||
|  | if (flutterVersionName == null) { | ||||||
|  |     flutterVersionName = '1.0' | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | def keystoreProperties = new Properties() | ||||||
|  | def keystorePropertiesFile = rootProject.file('key.properties') | ||||||
|  | if (keystorePropertiesFile.exists()) { | ||||||
|  |     keystoreProperties.load(new FileInputStream(keystorePropertiesFile)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | android { | ||||||
|  |     namespace "com.example.ollama" | ||||||
|  |     compileSdk flutter.compileSdkVersion | ||||||
|  |     ndkVersion flutter.ndkVersion | ||||||
|  | 
 | ||||||
|  |     dependenciesInfo { | ||||||
|  |         includeInApk = false | ||||||
|  |         includeInBundle = false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     compileOptions { | ||||||
|  |         sourceCompatibility JavaVersion.VERSION_1_8 | ||||||
|  |         targetCompatibility JavaVersion.VERSION_1_8 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     kotlinOptions { | ||||||
|  |         jvmTarget = '1.8' | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     sourceSets { | ||||||
|  |         main.java.srcDirs += 'src/main/kotlin' | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     defaultConfig { | ||||||
|  |         applicationId "com.freakurl.apps.ollama" | ||||||
|  |         // You can update the following values to match your application needs. | ||||||
|  |         // For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration. | ||||||
|  |         minSdkVersion flutter.minSdkVersion | ||||||
|  |         targetSdkVersion flutter.targetSdkVersion | ||||||
|  |         versionCode flutterVersionCode.toInteger() | ||||||
|  |         versionName flutterVersionName | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     signingConfigs { | ||||||
|  |         release { | ||||||
|  |             keyAlias keystoreProperties['keyAlias'] | ||||||
|  |             keyPassword keystoreProperties['keyPassword'] | ||||||
|  |             storeFile keystoreProperties['storeFile'] ? file(keystoreProperties['storeFile']) : null | ||||||
|  |             storePassword keystoreProperties['storePassword'] | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     buildTypes { | ||||||
|  |         release { | ||||||
|  |             // Signing with the debug keys for now, so `flutter run --release` works. | ||||||
|  |             signingConfig signingConfigs.release | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flutter { | ||||||
|  |     source '../..' | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | dependencies {} | ||||||
|  | @ -1,59 +0,0 @@ | ||||||
| import java.util.Properties |  | ||||||
| import java.io.FileInputStream |  | ||||||
| 
 |  | ||||||
| plugins { |  | ||||||
|     id("com.android.application") |  | ||||||
|     id("kotlin-android") |  | ||||||
|     // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. |  | ||||||
|     id("dev.flutter.flutter-gradle-plugin") |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| val keystoreProperties = Properties() |  | ||||||
| val keystorePropertiesFile = rootProject.file("key.properties") |  | ||||||
| if (keystorePropertiesFile.exists()) { |  | ||||||
|     keystoreProperties.load(FileInputStream(keystorePropertiesFile)) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| android { |  | ||||||
|     namespace = "com.freakurl.apps.ollama" |  | ||||||
|     compileSdk = flutter.compileSdkVersion |  | ||||||
|     ndkVersion = "27.0.12077973" // flutter.ndkVersion |  | ||||||
| 
 |  | ||||||
|     compileOptions { |  | ||||||
|         sourceCompatibility = JavaVersion.VERSION_11 |  | ||||||
|         targetCompatibility = JavaVersion.VERSION_11 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     kotlinOptions { |  | ||||||
|         jvmTarget = JavaVersion.VERSION_11.toString() |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     defaultConfig { |  | ||||||
|         applicationId = "com.freakurl.apps.ollama" |  | ||||||
|         // You can update the following values to match your application needs. |  | ||||||
|         // For more information, see: https://flutter.dev/to/review-gradle-config. |  | ||||||
|         minSdk = flutter.minSdkVersion |  | ||||||
|         targetSdk = flutter.targetSdkVersion |  | ||||||
|         versionCode = flutter.versionCode |  | ||||||
|         versionName = flutter.versionName |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     signingConfigs { |  | ||||||
|         create("release") { |  | ||||||
|             keyAlias = keystoreProperties["keyAlias"] as String |  | ||||||
|             keyPassword = keystoreProperties["keyPassword"] as String |  | ||||||
|             storeFile = keystoreProperties["storeFile"]?.let { file(it) } |  | ||||||
|             storePassword = keystoreProperties["storePassword"] as String |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     buildTypes { |  | ||||||
|         release { |  | ||||||
|             signingConfig = signingConfigs.getByName("release") |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| flutter { |  | ||||||
|     source = "../.." |  | ||||||
| } |  | ||||||
|  | @ -2,13 +2,11 @@ | ||||||
|     <application |     <application | ||||||
|         android:label="Ollama" |         android:label="Ollama" | ||||||
|         android:name="${applicationName}" |         android:name="${applicationName}" | ||||||
|         android:icon="@mipmap/ic_launcher" |         android:icon="@mipmap/ic_launcher"> | ||||||
|         android:enableOnBackInvokedCallback="true"> |  | ||||||
|         <activity |         <activity | ||||||
|             android:name=".MainActivity" |             android:name=".MainActivity" | ||||||
|             android:exported="true" |             android:exported="true" | ||||||
|             android:launchMode="singleTop" |             android:launchMode="singleTop" | ||||||
|             android:taskAffinity="" |  | ||||||
|             android:theme="@style/LaunchTheme" |             android:theme="@style/LaunchTheme" | ||||||
|             android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode" |             android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode" | ||||||
|             android:hardwareAccelerated="true" |             android:hardwareAccelerated="true" | ||||||
|  | @ -18,12 +16,12 @@ | ||||||
|                  while the Flutter UI initializes. After that, this theme continues |                  while the Flutter UI initializes. After that, this theme continues | ||||||
|                  to determine the Window background behind the Flutter UI. --> |                  to determine the Window background behind the Flutter UI. --> | ||||||
|             <meta-data |             <meta-data | ||||||
|               android:name="io.flutter.embedding.android.NormalTheme" |                 android:name="io.flutter.embedding.android.NormalTheme" | ||||||
|               android:resource="@style/NormalTheme" |                 android:resource="@style/NormalTheme" | ||||||
|               /> |             /> | ||||||
|             <intent-filter> |             <intent-filter> | ||||||
|                 <action android:name="android.intent.action.MAIN"/> |                 <action android:name="android.intent.action.MAIN" /> | ||||||
|                 <category android:name="android.intent.category.LAUNCHER"/> |                 <category android:name="android.intent.category.LAUNCHER" /> | ||||||
|             </intent-filter> |             </intent-filter> | ||||||
|         </activity> |         </activity> | ||||||
|         <!-- Don't delete the meta-data below. |         <!-- Don't delete the meta-data below. | ||||||
|  | @ -33,39 +31,25 @@ | ||||||
|             android:value="2" /> |             android:value="2" /> | ||||||
|     </application> |     </application> | ||||||
|     <!-- Required to query activities that can process text, see: |     <!-- Required to query activities that can process text, see: | ||||||
|          https://developer.android.com/training/package-visibility and |          https://developer.android.com/training/package-visibility?hl=en and | ||||||
|          https://developer.android.com/reference/android/content/Intent#ACTION_PROCESS_TEXT. |          https://developer.android.com/reference/android/content/Intent#ACTION_PROCESS_TEXT. | ||||||
| 
 | 
 | ||||||
|          In particular, this is used by the Flutter engine in io.flutter.plugin.text.ProcessTextPlugin. --> |          In particular, this is used by the Flutter engine in io.flutter.plugin.text.ProcessTextPlugin. --> | ||||||
|     <queries> |     <queries> | ||||||
|         <intent> |         <intent> | ||||||
|             <action android:name="android.intent.action.PROCESS_TEXT"/> |             <action android:name="android.intent.action.PROCESS_TEXT" /> | ||||||
|             <data android:mimeType="text/plain"/> |             <data android:mimeType="text/plain" /> | ||||||
|         </intent> |         </intent> | ||||||
|         <!-- check if url is valid --> |         <!-- check if url is valid --> | ||||||
|         <intent> |         <intent> | ||||||
|             <action android:name="android.intent.action.VIEW"/> |             <action android:name="android.intent.action.VIEW" /> | ||||||
|             <data android:scheme="http"/> |             <data android:scheme="http" /> | ||||||
|         </intent> |         </intent> | ||||||
|         <intent> |         <intent> | ||||||
|             <action android:name="android.intent.action.VIEW"/> |             <action android:name="android.intent.action.VIEW" /> | ||||||
|             <data android:scheme="https"/> |             <data android:scheme="https"/> | ||||||
|         </intent> |         </intent> | ||||||
|         <!-- check if url is valid end --> |         <!-- check if url is valid end --> | ||||||
|         <!-- voice mode --> |  | ||||||
|         <intent> |  | ||||||
|             <action android:name="android.speech.RecognitionService"/> |  | ||||||
|         </intent> |  | ||||||
|         <intent>   |  | ||||||
|             <action android:name="android.intent.action.TTS_SERVICE"/> |  | ||||||
|         </intent> |  | ||||||
|         <!-- voice mode end--> |  | ||||||
|     </queries> |     </queries> | ||||||
|     <uses-permission android:name="android.permission.INTERNET" /> |     <uses-permission android:name="android.permission.INTERNET" /> | ||||||
|     <!-- voice mode --> | </manifest> | ||||||
|     <uses-permission android:name="android.permission.RECORD_AUDIO"/> |  | ||||||
|     <uses-permission android:name="android.permission.BLUETOOTH"/> |  | ||||||
|     <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/> |  | ||||||
|     <uses-permission android:name="android.permission.BLUETOOTH_CONNECT"/> |  | ||||||
|     <!-- voice mode end --> |  | ||||||
| </manifest> |  | ||||||
|  | @ -0,0 +1,5 @@ | ||||||
|  | package com.example.ollama | ||||||
|  | 
 | ||||||
|  | import io.flutter.embedding.android.FlutterActivity | ||||||
|  | 
 | ||||||
|  | class MainActivity: FlutterActivity() | ||||||
|  | @ -1,5 +0,0 @@ | ||||||
| package com.freakurl.apps.ollama |  | ||||||
| 
 |  | ||||||
| import io.flutter.embedding.android.FlutterActivity |  | ||||||
| 
 |  | ||||||
| class MainActivity : FlutterActivity() |  | ||||||
|  | @ -0,0 +1,18 @@ | ||||||
|  | allprojects { | ||||||
|  |     repositories { | ||||||
|  |         google() | ||||||
|  |         mavenCentral() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | rootProject.buildDir = '../build' | ||||||
|  | subprojects { | ||||||
|  |     project.buildDir = "${rootProject.buildDir}/${project.name}" | ||||||
|  | } | ||||||
|  | subprojects { | ||||||
|  |     project.evaluationDependsOn(':app') | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | tasks.register("clean", Delete) { | ||||||
|  |     delete rootProject.buildDir | ||||||
|  | } | ||||||
|  | @ -1,21 +0,0 @@ | ||||||
| allprojects { |  | ||||||
|     repositories { |  | ||||||
|         google() |  | ||||||
|         mavenCentral() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| val newBuildDir: Directory = rootProject.layout.buildDirectory.dir("../../build").get() |  | ||||||
| rootProject.layout.buildDirectory.value(newBuildDir) |  | ||||||
| 
 |  | ||||||
| subprojects { |  | ||||||
|     val newSubprojectBuildDir: Directory = newBuildDir.dir(project.name) |  | ||||||
|     project.layout.buildDirectory.value(newSubprojectBuildDir) |  | ||||||
| } |  | ||||||
| subprojects { |  | ||||||
|     project.evaluationDependsOn(":app") |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| tasks.register<Delete>("clean") { |  | ||||||
|     delete(rootProject.layout.buildDirectory) |  | ||||||
| } |  | ||||||
|  | @ -1,3 +1,3 @@ | ||||||
| org.gradle.jvmargs=-Xmx8G -XX:MaxMetaspaceSize=4G -XX:ReservedCodeCacheSize=512m -XX:+HeapDumpOnOutOfMemoryError | org.gradle.jvmargs=-Xmx4G | ||||||
| android.useAndroidX=true | android.useAndroidX=true | ||||||
| android.enableJetifier=true | android.enableJetifier=true | ||||||
|  |  | ||||||
|  | @ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.10.2-all.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-7.6.3-all.zip | ||||||
|  |  | ||||||
|  | @ -0,0 +1,26 @@ | ||||||
|  | pluginManagement { | ||||||
|  |     def flutterSdkPath = { | ||||||
|  |         def properties = new Properties() | ||||||
|  |         file("local.properties").withInputStream { properties.load(it) } | ||||||
|  |         def flutterSdkPath = properties.getProperty("flutter.sdk") | ||||||
|  |         assert flutterSdkPath != null, "flutter.sdk not set in local.properties" | ||||||
|  |         return flutterSdkPath | ||||||
|  |     } | ||||||
|  |     settings.ext.flutterSdkPath = flutterSdkPath() | ||||||
|  | 
 | ||||||
|  |     includeBuild("${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle") | ||||||
|  | 
 | ||||||
|  |     repositories { | ||||||
|  |         google() | ||||||
|  |         mavenCentral() | ||||||
|  |         gradlePluginPortal() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | plugins { | ||||||
|  |     id "dev.flutter.flutter-plugin-loader" version "1.0.0" | ||||||
|  |     id "com.android.application" version "7.3.0" apply false | ||||||
|  |     id "org.jetbrains.kotlin.android" version "1.7.10" apply false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | include ":app" | ||||||
|  | @ -1,25 +0,0 @@ | ||||||
| pluginManagement { |  | ||||||
|     val flutterSdkPath = run { |  | ||||||
|         val properties = java.util.Properties() |  | ||||||
|         file("local.properties").inputStream().use { properties.load(it) } |  | ||||||
|         val flutterSdkPath = properties.getProperty("flutter.sdk") |  | ||||||
|         require(flutterSdkPath != null) { "flutter.sdk not set in local.properties" } |  | ||||||
|         flutterSdkPath |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     includeBuild("$flutterSdkPath/packages/flutter_tools/gradle") |  | ||||||
| 
 |  | ||||||
|     repositories { |  | ||||||
|         google() |  | ||||||
|         mavenCentral() |  | ||||||
|         gradlePluginPortal() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| plugins { |  | ||||||
|     id("dev.flutter.flutter-plugin-loader") version "1.0.0" |  | ||||||
|     id("com.android.application") version "8.7.0" apply false |  | ||||||
|     id("org.jetbrains.kotlin.android") version "1.8.22" apply false |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| include(":app") |  | ||||||
| Before Width: | Height: | Size: 1.2 MiB | 
| Before Width: | Height: | Size: 17 KiB | 
| Before Width: | Height: | Size: 53 KiB | 
| Before Width: | Height: | Size: 92 KiB | 
| Before Width: | Height: | Size: 45 KiB | 
| Before Width: | Height: | Size: 81 KiB | 
| Before Width: | Height: | Size: 82 KiB | 
| Before Width: | Height: | Size: 78 KiB | 
| Before Width: | Height: | Size: 87 KiB | 
| Before Width: | Height: | Size: 154 KiB | 
| Before Width: | Height: | Size: 137 KiB | 
| Before Width: | Height: | Size: 83 KiB | 
| Before Width: | Height: | Size: 75 KiB | 
| Before Width: | Height: | Size: 157 KiB | 
| Before Width: | Height: | Size: 73 KiB | 
| Before Width: | Height: | Size: 109 KiB | 
| Before Width: | Height: | Size: 74 KiB | 
| Before Width: | Height: | Size: 69 KiB | 
| Before Width: | Height: | Size: 101 KiB | 
| Before Width: | Height: | Size: 104 KiB | 
| Before Width: | Height: | Size: 122 KiB | 
| Before Width: | Height: | Size: 121 KiB | 
| Before Width: | Height: | Size: 139 KiB | 
| Before Width: | Height: | Size: 88 KiB | 
| Before Width: | Height: | Size: 125 KiB | 
| Before Width: | Height: | Size: 116 KiB | 
| Before Width: | Height: | Size: 39 KiB | 
| Before Width: | Height: | Size: 151 KiB | 
| Before Width: | Height: | Size: 62 KiB | 
| Before Width: | Height: | Size: 140 KiB | 
| Before Width: | Height: | Size: 56 KiB | 
| Before Width: | Height: | Size: 60 KiB | 
| Before Width: | Height: | Size: 513 KiB | 
| Before Width: | Height: | Size: 643 KiB | 
| Before Width: | Height: | Size: 39 KiB | 
| Before Width: | Height: | Size: 57 KiB | 
| Before Width: | Height: | Size: 93 KiB | 
| Before Width: | Height: | Size: 181 KiB | 
| Before Width: | Height: | Size: 25 KiB | 
| Before Width: | Height: | Size: 32 KiB | 
| Before Width: | Height: | Size: 169 KiB | 
| Before Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 39 KiB | 
| Before Width: | Height: | Size: 39 KiB | 
| After Width: | Height: | Size: 38 KiB | 
| After Width: | Height: | Size: 118 KiB | 
| After Width: | Height: | Size: 59 KiB | 
| After Width: | Height: | Size: 146 KiB | 
| After Width: | Height: | Size: 110 KiB | 
| After Width: | Height: | Size: 212 KiB | 
| After Width: | Height: | Size: 37 KiB | 
| After Width: | Height: | Size: 118 KiB | 
| After Width: | Height: | Size: 37 KiB | 
| After Width: | Height: | Size: 118 KiB | 
| After Width: | Height: | Size: 57 KiB | 
| After Width: | Height: | Size: 152 KiB | 
| After Width: | Height: | Size: 21 KiB | 
| After Width: | Height: | Size: 14 KiB | 
| After Width: | Height: | Size: 18 KiB | 
| After Width: | Height: | Size: 56 KiB | 
| After Width: | Height: | Size: 19 KiB | 
| After Width: | Height: | Size: 14 KiB | 
| After Width: | Height: | Size: 34 KiB | 
| After Width: | Height: | Size: 40 KiB | 
| After Width: | Height: | Size: 11 KiB | 
| Before Width: | Height: | Size: 62 KiB | 
| Before Width: | Height: | Size: 45 KiB | 
| Before Width: | Height: | Size: 73 KiB After Width: | Height: | Size: 64 KiB | 
| Before Width: | Height: | Size: 69 KiB After Width: | Height: | Size: 64 KiB | 
| Before Width: | Height: | Size: 160 KiB After Width: | Height: | Size: 152 KiB | 
| Before Width: | Height: | Size: 166 KiB After Width: | Height: | Size: 160 KiB | 
| Before Width: | Height: | Size: 98 KiB After Width: | Height: | Size: 120 KiB | 
| Before Width: | Height: | Size: 98 KiB After Width: | Height: | Size: 120 KiB | 
|  | @ -1,3 +0,0 @@ | ||||||
| description: This file stores settings for Dart & Flutter DevTools. |  | ||||||
| documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states |  | ||||||
| extensions: |  | ||||||
| After Width: | Height: | Size: 35 KiB | 
| After Width: | Height: | Size: 69 KiB | 
| After Width: | Height: | Size: 100 KiB | 
| After Width: | Height: | Size: 170 KiB |