├── gradle.properties ├── assets └── xposed_init ├── ic_launcher-web.png ├── res ├── xml │ └── backupscheme.xml ├── mipmap-hdpi │ └── ic_launcher.png ├── mipmap-mdpi │ └── ic_launcher.png ├── mipmap-xhdpi │ └── ic_launcher.png ├── mipmap-xxhdpi │ └── ic_launcher.png ├── mipmap-xxxhdpi │ └── ic_launcher.png ├── values │ ├── colors.xml │ ├── dimens.xml │ ├── styles.xml │ └── strings.xml └── layout │ └── activity_settings.xml ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── java └── com │ └── sabik │ └── assistantenabler │ ├── ChangeSettingsReceiver.java │ ├── Settings.java │ └── AssistantEnabler.java ├── LICENSE ├── README.md ├── AndroidManifest.xml ├── gradlew.bat ├── .gitignore ├── gradlew └── AssistantEnabler.iml /gradle.properties: -------------------------------------------------------------------------------- 1 | org.gradle.jvmargs=-Xmx2048M 2 | -------------------------------------------------------------------------------- /assets/xposed_init: -------------------------------------------------------------------------------- 1 | com.sabik.assistantenabler.AssistantEnabler -------------------------------------------------------------------------------- /ic_launcher-web.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/ic_launcher-web.png -------------------------------------------------------------------------------- /res/xml/backupscheme.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Sabissimo/AssistantEnabler/HEAD/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #03A9F4 4 | #0288D1 5 | #e7524b 6 | 7 | -------------------------------------------------------------------------------- /res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | # Tue Oct 4 00:20:00 EEST 2016 2 | distributionBase=GRADLE_USER_HOME 3 | distributionPath=wrapper/dists 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | distributionUrl=https\://services.gradle.org/distributions/gradle-3.1-all.zip 7 | -------------------------------------------------------------------------------- /res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /java/com/sabik/assistantenabler/ChangeSettingsReceiver.java: -------------------------------------------------------------------------------- 1 | package com.sabik.assistantenabler; 2 | 3 | import android.content.BroadcastReceiver; 4 | import android.content.Context; 5 | import android.content.Intent; 6 | import android.content.SharedPreferences; 7 | import android.os.Bundle; 8 | 9 | public class ChangeSettingsReceiver extends BroadcastReceiver { 10 | public ChangeSettingsReceiver() { 11 | } 12 | 13 | @Override 14 | public void onReceive(Context context, Intent intent) { 15 | 16 | final SharedPreferences prefs = context.getSharedPreferences("preferences",Context.MODE_WORLD_READABLE); 17 | final SharedPreferences.Editor editor = prefs.edit(); 18 | 19 | if(intent.getAction().equals("com.sabik.assistantenabler.CHANGE_SETTINGS")) 20 | { 21 | Bundle extras = intent.getExtras(); 22 | String settingName = extras.getString("setting"); 23 | Boolean settingValue = extras.getBoolean("value", false); 24 | if(settingName != null) 25 | switch(settingName){ 26 | case "googleNowEnabled": 27 | case "assistantEnabled": 28 | case "enableOKGoogleEverywhere": 29 | editor.putBoolean(settingName,settingValue); 30 | editor.apply(); 31 | } 32 | } 33 | } 34 | } 35 | -------------------------------------------------------------------------------- /res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Assistant Enabler 5 | Enable Google Assistant in Marshmallow ROMs 6 | Enable Assistant 7 | Hide from launcher 8 | Hide application icon from launcher 9 | Enable Google Assistant 10 | Enable Google Now (requires restart) 11 | Enable Google Now in not eligible countries 12 | Donate 13 | If you would like to support this project\'s further development, the creator of this project or the continuous maintenance of this project, feel free to donate. 14 | Enable OK Google everywhere 15 | Enable OK Google everywhere, even with screen off (might cause some battery draining). After changing this, plug/unplug your charger for setting to kick in. 16 | 17 | 18 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2016, Sabik 2 | All rights reserved. 3 | 4 | Redistribution and use in source and binary forms, with or without 5 | modification, are permitted provided that the following conditions are met: 6 | 7 | * Redistributions of source code must retain the above copyright notice, this 8 | list of conditions and the following disclaimer. 9 | 10 | * Redistributions in binary form must reproduce the above copyright notice, 11 | this list of conditions and the following disclaimer in the documentation 12 | and/or other materials provided with the distribution. 13 | 14 | * Neither the name of AssistantEnabler nor the names of its 15 | contributors may be used to endorse or promote products derived from 16 | this software without specific prior written permission. 17 | 18 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 19 | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20 | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 21 | DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 22 | FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 23 | DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 24 | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 25 | CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 26 | OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 27 | OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 | 29 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Assistant Enabler 2 | 3 | Xposed module: Enable Google Assistant in Marshmallow ROMs 4 | 5 | Enable/disable Google Assistant without restart 6 | Bonus feature: Enable Google Now in unsupported countries* 7 | 8 | ## How to install 9 | 10 | Simply grab the module from [the Xposed Module Repository](http://repo.xposed.info/module/com.sabik.assistantenabler). 11 | 12 | To make a build build on your own device from source, feel free to clone this repository and use Gradle for that. You can read more about how to build locally on [Android Developers](https://developer.android.com/tools/building/building-cmdline.html) or just Google around. 13 | 14 | 15 | #After 6.10 update "OK Google" detection is working as intended without any interference, so just go ahead in google settings and configure voice there, it works now! 16 | #For versions below 6.10 use following guide: 17 | #To use "OK Google" hotword you need to do following: 18 | * 1. Update Google App to the latest version (at least 6.8 beta) 19 | * 2. Disable assistant in module 20 | * 3. Configure "OK Google" detection and voice pattern 21 | * 4. Enable assistant in module 22 | * 5. Say "OK Google" 23 | * 6. .... 24 | * 7. Profit! 25 | 26 | # Enable "OK Google" with screen off (even when not charging) 27 | * After enabling it in module settings, plug/unplug your charger for settings to kick in 28 | 29 | # In order to enable Google Now in unsupported country, these steps are required: 30 | * 1) Enable Assistant Enabler module in Xposed Installer 31 | * 2) In Assistant Enabler settings, enable "Enable Google Now" 32 | * 3) Reboot (after rebooting your phone carrier will be identified as US Verizon) 33 | * 4) Either remove your sim, or put your phone in airplane mode 34 | * 5) Enable Wifi 35 | * 6) If google account is added to the phone already - remove it 36 | * 7) Launch google app and login to your google account 37 | * 8) Opt-in to Google Now 38 | * 9) PROFIT! 39 | 40 | P.S. After these steps you can disable "Enable Google Now" setting in Assistant Enabler module, if you don't want your carrier to be identified as VZW. Google Now will continue working. -------------------------------------------------------------------------------- /AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | 14 | 17 | 21 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @if "%DEBUG%" == "" @echo off 2 | @rem ########################################################################## 3 | @rem 4 | @rem Gradle startup script for Windows 5 | @rem 6 | @rem ########################################################################## 7 | 8 | @rem Set local scope for the variables with windows NT shell 9 | if "%OS%"=="Windows_NT" setlocal 10 | 11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 12 | set DEFAULT_JVM_OPTS= 13 | 14 | set DIRNAME=%~dp0 15 | if "%DIRNAME%" == "" set DIRNAME=. 16 | set APP_BASE_NAME=%~n0 17 | set APP_HOME=%DIRNAME% 18 | 19 | @rem Find java.exe 20 | if defined JAVA_HOME goto findJavaFromJavaHome 21 | 22 | set JAVA_EXE=java.exe 23 | %JAVA_EXE% -version >NUL 2>&1 24 | if "%ERRORLEVEL%" == "0" goto init 25 | 26 | echo. 27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 28 | echo. 29 | echo Please set the JAVA_HOME variable in your environment to match the 30 | echo location of your Java installation. 31 | 32 | goto fail 33 | 34 | :findJavaFromJavaHome 35 | set JAVA_HOME=%JAVA_HOME:"=% 36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 37 | 38 | if exist "%JAVA_EXE%" goto init 39 | 40 | echo. 41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 42 | echo. 43 | echo Please set the JAVA_HOME variable in your environment to match the 44 | echo location of your Java installation. 45 | 46 | goto fail 47 | 48 | :init 49 | @rem Get command-line arguments, handling Windowz variants 50 | 51 | if not "%OS%" == "Windows_NT" goto win9xME_args 52 | if "%@eval[2+2]" == "4" goto 4NT_args 53 | 54 | :win9xME_args 55 | @rem Slurp the command line arguments. 56 | set CMD_LINE_ARGS= 57 | set _SKIP=2 58 | 59 | :win9xME_args_slurp 60 | if "x%~1" == "x" goto execute 61 | 62 | set CMD_LINE_ARGS=%* 63 | goto execute 64 | 65 | :4NT_args 66 | @rem Get arguments from the 4NT Shell from JP Software 67 | set CMD_LINE_ARGS=%$ 68 | 69 | :execute 70 | @rem Setup the command line 71 | 72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if "%ERRORLEVEL%"=="0" goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 85 | exit /b 1 86 | 87 | :mainEnd 88 | if "%OS%"=="Windows_NT" endlocal 89 | 90 | :omega 91 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | ################# 2 | ## Eclipse 3 | ################# 4 | 5 | *.pydevproject 6 | .project 7 | .metadata 8 | bin/ 9 | tmp/ 10 | *.tmp 11 | *.bak 12 | *.swp 13 | *~.nib 14 | local.properties 15 | .idea 16 | .gradle 17 | .classpath 18 | .settings/ 19 | .loadpath 20 | 21 | # External tool builders 22 | .externalToolBuilders/ 23 | 24 | # Locally stored "Eclipse launch configurations" 25 | *.launch 26 | 27 | # CDT-specific 28 | .cproject 29 | 30 | # PDT-specific 31 | .buildpath 32 | 33 | 34 | ################# 35 | ## Visual Studio 36 | ################# 37 | 38 | ## Ignore Visual Studio temporary files, build results, and 39 | ## files generated by popular Visual Studio add-ons. 40 | 41 | # User-specific files 42 | *.suo 43 | *.user 44 | *.sln.docstates 45 | 46 | # Build results 47 | 48 | [Dd]ebug/ 49 | [Rr]elease/ 50 | x64/ 51 | build/ 52 | [Bb]in/ 53 | [Oo]bj/ 54 | 55 | # MSTest test Results 56 | [Tt]est[Rr]esult*/ 57 | [Bb]uild[Ll]og.* 58 | 59 | *_i.c 60 | *_p.c 61 | *.ilk 62 | *.meta 63 | *.obj 64 | *.pch 65 | *.pdb 66 | *.pgc 67 | *.pgd 68 | *.rsp 69 | *.sbr 70 | *.tlb 71 | *.tli 72 | *.tlh 73 | *.tmp 74 | *.tmp_proj 75 | *.log 76 | *.vspscc 77 | *.vssscc 78 | .builds 79 | *.pidb 80 | *.log 81 | *.scc 82 | 83 | # Visual C++ cache files 84 | ipch/ 85 | *.aps 86 | *.ncb 87 | *.opensdf 88 | *.sdf 89 | *.cachefile 90 | 91 | # Visual Studio profiler 92 | *.psess 93 | *.vsp 94 | *.vspx 95 | 96 | # Guidance Automation Toolkit 97 | *.gpState 98 | 99 | # ReSharper is a .NET coding add-in 100 | _ReSharper*/ 101 | *.[Rr]e[Ss]harper 102 | 103 | # TeamCity is a build add-in 104 | _TeamCity* 105 | 106 | # DotCover is a Code Coverage Tool 107 | *.dotCover 108 | 109 | # NCrunch 110 | *.ncrunch* 111 | .*crunch*.local.xml 112 | 113 | # Installshield output folder 114 | [Ee]xpress/ 115 | 116 | # DocProject is a documentation generator add-in 117 | DocProject/buildhelp/ 118 | DocProject/Help/*.HxT 119 | DocProject/Help/*.HxC 120 | DocProject/Help/*.hhc 121 | DocProject/Help/*.hhk 122 | DocProject/Help/*.hhp 123 | DocProject/Help/Html2 124 | DocProject/Help/html 125 | 126 | # Click-Once directory 127 | publish/ 128 | 129 | # Publish Web Output 130 | *.Publish.xml 131 | *.pubxml 132 | *.publishproj 133 | 134 | # NuGet Packages Directory 135 | ## TODO: If you have NuGet Package Restore enabled, uncomment the next line 136 | #packages/ 137 | 138 | # Windows Azure Build Output 139 | csx 140 | *.build.csdef 141 | 142 | # Windows Store app package directory 143 | AppPackages/ 144 | 145 | # Others 146 | sql/ 147 | *.Cache 148 | ClientBin/ 149 | [Ss]tyle[Cc]op.* 150 | ~$* 151 | *~ 152 | *.dbmdl 153 | *.[Pp]ublish.xml 154 | *.pfx 155 | *.publishsettings 156 | 157 | # RIA/Silverlight projects 158 | Generated_Code/ 159 | 160 | # Backup & report files from converting an old project file to a newer 161 | # Visual Studio version. Backup files are not needed, because we have git ;-) 162 | _UpgradeReport_Files/ 163 | Backup*/ 164 | UpgradeLog*.XML 165 | UpgradeLog*.htm 166 | 167 | # SQL Server files 168 | App_Data/*.mdf 169 | App_Data/*.ldf 170 | 171 | ############# 172 | ## Windows detritus 173 | ############# 174 | 175 | # Windows image file caches 176 | Thumbs.db 177 | ehthumbs.db 178 | 179 | # Folder config file 180 | Desktop.ini 181 | 182 | # Recycle Bin used on file shares 183 | $RECYCLE.BIN/ 184 | 185 | # Mac crap 186 | .DS_Store 187 | 188 | 189 | ############# 190 | ## Python 191 | ############# 192 | 193 | *.py[cod] 194 | 195 | # Packages 196 | *.egg 197 | *.egg-info 198 | dist/ 199 | build/ 200 | eggs/ 201 | parts/ 202 | var/ 203 | sdist/ 204 | develop-eggs/ 205 | .installed.cfg 206 | 207 | # Installer logs 208 | pip-log.txt 209 | 210 | # Unit test / coverage reports 211 | .coverage 212 | .tox 213 | 214 | #Translations 215 | *.mo 216 | 217 | #Mr Developer 218 | .mr.developer.cfg 219 | -------------------------------------------------------------------------------- /java/com/sabik/assistantenabler/Settings.java: -------------------------------------------------------------------------------- 1 | package com.sabik.assistantenabler; 2 | 3 | import android.content.ComponentName; 4 | import android.content.Context; 5 | import android.content.Intent; 6 | import android.content.SharedPreferences; 7 | import android.content.pm.PackageManager; 8 | import android.net.Uri; 9 | import android.support.v7.app.AppCompatActivity; 10 | import android.os.Bundle; 11 | import android.view.View; 12 | import android.widget.Button; 13 | import android.widget.CompoundButton; 14 | import android.widget.Switch; 15 | 16 | public class Settings extends AppCompatActivity { 17 | 18 | public Context context; 19 | 20 | @Override 21 | protected void onCreate(Bundle savedInstanceState) { 22 | super.onCreate(savedInstanceState); 23 | setContentView(R.layout.activity_settings); 24 | 25 | final SharedPreferences prefs = getSharedPreferences("preferences",MODE_WORLD_READABLE); 26 | final SharedPreferences.Editor editor = prefs.edit(); 27 | context = getApplicationContext(); 28 | Switch assistantEnabled = (Switch) findViewById(R.id.assistantEnabled); 29 | Switch enableOKGoogleEverywhere = (Switch) findViewById(R.id.enableOKGoogleEverywhere); 30 | Switch googleNowEnabled = (Switch) findViewById(R.id.googleNowEnabled); 31 | Switch hiddenIcon = (Switch) findViewById(R.id.hiddenIcon); 32 | Button donateButton = (Button) findViewById(R.id.donateButton); 33 | 34 | if (googleNowEnabled!=null) { 35 | googleNowEnabled.setChecked(prefs.getBoolean("googleNowEnabled",false)); 36 | googleNowEnabled.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 37 | public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 38 | 39 | editor.putBoolean("googleNowEnabled", isChecked); 40 | editor.apply(); 41 | 42 | } 43 | }); 44 | } 45 | 46 | if (assistantEnabled!=null) { 47 | assistantEnabled.setChecked(prefs.getBoolean("assistantEnabled",true)); 48 | assistantEnabled.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 49 | public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 50 | 51 | editor.putBoolean("assistantEnabled", isChecked); 52 | editor.apply(); 53 | 54 | } 55 | }); 56 | } 57 | 58 | if (enableOKGoogleEverywhere!=null) { 59 | enableOKGoogleEverywhere.setChecked(prefs.getBoolean("enableOKGoogleEverywhere",false)); 60 | enableOKGoogleEverywhere.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 61 | public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 62 | 63 | editor.putBoolean("enableOKGoogleEverywhere", isChecked); 64 | editor.apply(); 65 | 66 | } 67 | }); 68 | } 69 | 70 | if (hiddenIcon!=null) { 71 | hiddenIcon.setChecked(prefs.getBoolean("hiddenIcon",false)); 72 | hiddenIcon.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 73 | public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 74 | 75 | PackageManager packageManager = context.getPackageManager(); 76 | int state = isChecked ? PackageManager.COMPONENT_ENABLED_STATE_DISABLED 77 | : PackageManager.COMPONENT_ENABLED_STATE_ENABLED; 78 | String settings = BuildConfig.APPLICATION_ID + ".Preferences"; 79 | ComponentName alias = new ComponentName(context, settings); 80 | packageManager.setComponentEnabledSetting(alias, state, 81 | PackageManager.DONT_KILL_APP); 82 | 83 | editor.putBoolean("hiddenIcon", isChecked); 84 | editor.apply(); 85 | 86 | } 87 | }); 88 | } 89 | 90 | if (donateButton!=null){ 91 | donateButton.setOnClickListener(new View.OnClickListener() { 92 | 93 | public void onClick(View v) { 94 | startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("https://www.paypal.me/Sabissimo"))); 95 | } 96 | }); 97 | } 98 | 99 | } 100 | } 101 | -------------------------------------------------------------------------------- /res/layout/activity_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 12 | 13 | 21 | 22 | 30 | 31 | 40 | 41 | 49 | 50 | 59 | 60 | 68 | 69 | 78 | 79 | 87 | 88 | 97 | 98 |