Rename all file to kebab-case

This commit is contained in:
Marcello 2022-05-13 18:33:39 +02:00
parent ffc085d78a
commit 68ca4e4f86
117 changed files with 0 additions and 1260 deletions

View file

@ -1,438 +0,0 @@
# Android App Activity.kt
## Logging
```kotlin
Log.i("tag", "logValue") //info log
Log.d("tag", "logValue") //debug log
Log.w("tag", "logValue") //warning log
Log.e("tag", "logValue") //error log
Log.c("tag", "logValue") //critical log
```
## Activity Life Cycle
![Life Cycle](../.images/android_activity-lifecycle.png)
```kotlin
package com.its.<appname>
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
class MainActivity : AppCompatActivity() {
//entry point of the activity
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.<activity_xml>)
}
override fun onStart() {
}
override fun onResume() {
}
override fun onPause() {
}
override fun onStop() {
}
override fun onRestart() {
}
override fun onDestroy() {
}
}
```
### Passing data between activities
In *laughing* activity:
```kotlin
private fun openActivity() {
//target to open Intent(opener, opened)
val intent = Intent(this, Activity::class.java)
//pass data to launched activity
intent.putExtra("identifier", value)
startActivity(intent) //launch another activity
}
```
In *launched* activity:
```kotlin
val identifier = intent.get<Type>Extra("identifier")
```
## Hooks
### Resources Hooks
```kotlin
R.<resourceType>.<resourceName> //access to resource
ContextCompat.getColor(this, colorResource) //extract color from resources
getString(stringResource) //extract string from resources
```
### XML hooks
Changes in xml made in kotlin are applied after the app is drawn thus overwriting instructions in xml.
In `activity.xml`:
```xml
<View
android:id="@+id/<id>"/>
```
in `Activity.kt`:
```kotlin
var element = findViewById(R.id.<id>) //old method
<id>.popery = value //access and modify view contents
```
## Activity Components
### Snackbar
Component derived from material design. If using old API material design dependency must be set in gradle.
In `build.gradle (Module:app)`:
```gradle
dependencies {
implementation 'com.google.android.material:material:<sem_ver>'
}
```
In `Activity.kt`:
```kotlin
import com.google.android.material.snackbar.Snackbar
Snackbar
.make(activityID, message, Snackbar.TIME_ALIVE) //create snackbar
.setAction("Button Name", { action }) //add button to snackbar
.show()
```
## Opening External Content
### Opening URLs
```kotlin
val url = "https://www.google.com"
val intent = Intent(Intent.ACTION_VIEW)
intent.setData(Uri.parse(url))
startActivity(intent)
```
### Sharing Content
```kotlin
val intent = Intent(Intent.ACTION_SEND)
intent.setType("text/plain") //specifying shared content type
intent.putExtra(Intent.EXTRA_MAIL, "mail@address") //open mail client and pre-compile field if share w/ mail
intent.putExtra(Intent.EXTRA_SUBJECT, "subject")
intent.putExtra(Intent.EXTRA_TEXT, "text") //necessary since type is text
startActivity(Intent.startChooser(intent, "chooser title")) //let user choose the share app
```
### App Google Maps
[Documentation](https://developers.google.com/maps/documentation/urls/android-intents)
```kotlin
val uri = Uri.parse("geo: <coordinates>")
val intent = Intent(Intent.ACTION_VIEW, uri)
intent.setPackage("com.google.android.apps.maps") //app to be opened
startActivity(intent)
```
### Make a call (wait for user)
Preset phone number for the call, user needs to press call button to initiate dialing.
```kotlin
fun call() {
val intent = Intent(Intent.ACTION_DIAL)
intent.setData(Uri.parse("tel: <phone number>"))
startActivity(intent)
}
```
### Make a call (directly)
In `AndroidManifest.xml`:
```xml
<uses-permission android:name="android.permission.CALL_PHONE" />
```
```kotlin
//https://developer.android.com/training/permissions/requesting
//intercept OS response to permission popup
override fun onRequestPermissionResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
}
fun checkCallPermission() {
//check if permission to make a call has been granted
if (ContextCompact.checkSelfPermission(context!!, android.Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED) {
// if permission has not been granted request it (opens OS popup, no listener available)
// request code needs to be specific for the permission
ActivityCompat.requestPermissions(context!!, arrayOf(android.Manifest.permission.CALL_PHONE), requestCode)
} else {
call() //if permission has been already given
}
}
@SuppressLint("MissingPermission") // suppress warning of unhandled permission (handled in checkCallPermission)
fun call() {
val intent = Intent(Intent.ACTION_DIAL, Uri.parse("tel: <phone number>"))
startActivity(intent)
}
```
## Lists (with RecyclerView)
A [LayoutManager][1] is responsible for measuring and positioning item views within a `RecyclerView` as well as determining the policy for when to recycle item views that are no longer visible to the user.
By changing the `LayoutManager` a `RecyclerView` can be used to implement a standard vertically *scrolling list*, a *uniform grid*, *staggered grids*, *horizontally scrolling collections* and more.
Several stock layout managers are provided for general use.
[Adapters][2] provide a binding from an app-specific data set to views that are displayed within a `RecyclerView`.
[1]:https://developer.android.com/reference/kotlin/androidx/recyclerview/widget/RecyclerView.LayoutManager
[2]:https://developer.android.com/reference/kotlin/androidx/recyclerview/widget/RecyclerView.Adapter
```kotlin
var array: ArrayList<T>? = null //create ArrayList of elements to be displayed
var adapter: RecyclerViewItemAdapter? = null //create adapter to draw the list in the Activity
array.add(item) //add item to ArrayList
// create LayoutManager for the recyclerView
val layoutManager = <ViewGroup>LayoutManager(context, <ViewGroup>LayoutManager.VERTICAL, reverseLayout: Bool)
recycleView.LayoutManager = layoutManager // assign LayoutManager for the recyclerView
// handle adapter var containing null value
if (array != null) {
adapter = RecyclerViewItemAdapter(array!!) // valorize adapter with a adapter object
recyclerView.adapter = adapter // assign adapter to the recyclerView
}
// add or remove item
// tell the adapter that something is changed
adapter?.notifyDataSetChanged()
```
## WebView
[WebView Docs](https://developerandroid.com/reference/android/webkit/WebView)
```kotlin
webView.webViewClient = object : WebViewClient() {
// avoid opening browsed by default, open webview instead
override fun shouldOverrideUrlLoading(view: WebView?, url: String?): Boolean {
view?.loadUrl(url) // handle every url
return true
}
override fun shouldOverrideUrlLoading(view: WebView?, url: String?): Boolean {
// stay in WebView until website changes
if (Uri.parse(url).host == "www.website.domain") {
return false
}
Intent(Intent.ACTION_VIEW, Uri.parse(url).apply){
startActivity(this) // open browser/app when the website changes to external URL
}
return true
}
webView.settings.javaScriptEnabled = true // enable javascript
webView.addJavaScriptInterface(webAppInterface(this, "Android")) // create bridge between js on website an app
// if in webView use android back butto to go to previous webpage (if possible)
override fun onKeyDown(keyCode: Int, event: KeyEvent?) :Boolean {
if (keyCode == KeyEvent.KEYCODE_BACK && webVew,canGoBack()) { // if previous url exists & back button pressed
webView.goBack() // go to previous URL
return true
}
}
}
```
## Web Requests (Using Volley)
[Volley Docs](https://developer.android.com/training/volley)
### Import & Permissions
Import in `build.gradle`:
```kotlin
implementation 'com.android.volley:volley:1.1.1'
```
Permissions in `AndroidManifest.xml`:
```xml
<uses-permission android:name="android.permission.INTERNET" />
```
### Make the request
Subsequent requests should be delayed to avoid allowing the user to make too frequent requests.
```kotlin
private lateinit var queue: RequestQueue
override fun onCreate(savedInstanceState: Bundle?){
queue = Volley.newRequestQueue(context)
}
// response is a String
private fun simpleRequest() {
var url = "www.website.domain"
var stringRequest = StringRequest(Request.Method.GET, url, Response.Listener<String> {
Log.d()
},
Response.ErrorListener{
Log.e()
})
stringRequest.TAG = "TAG" // assign a tag to the request
queue.add(stringRequest) // add request to the queue
}
// response is JSON Object
private fun getJsonObjectRequest(){
// GET -> jsonRequest = null
// POST -> jsonRequest = JsonObject
var stringRequest = JsonObjectRequest(Request.Method.GET, url, jsonRequest, Response.Listener<String> {
Log.d()
// read end use JSON
},
Response.ErrorListener{
Log.e()
})
queue.add(stringRequest) // add request to the queue
}
// response is array of JSON objects
private fun getJSONArrayRequest(){
// GET -> jsonRequest = null
// POST -> jsonRequest = JsonObject
var stringRequest = JsonArrayRequest(Request.Method.GET, url, jsonRequest, Response.Listener<String> {
Log.d()
// read end use JSON
},
Response.ErrorListener{
Log.e()
})
queue.add(stringRequest) // add request to the queue
}
override fun onStop() {
super.onStop()
queue?.cancelAll("TAG") // delete all request with a particular tag when the activity is closed (avoid crash)
}
```
### Parse JSON Request
```kotlin
Response.Listener { response ->
var value = response.getSting("key")
}
```
## Data Persistance
### Singleton
Object instantiated during app init and is destroyed only on app closing. It can be used for data persistance since is not affected by the destruction of an activity.
```kotlin
// Context: Interface to global information about an application environment.
class Singleton constructor(context: Context) {
companion object {
@Volatile
private var INSTANCE: Singleton? = null
// synchronized makes sure that all instances of the singleton are actually the only existing one
fun getInstance(context: Context) = INSTANCE ?: synchronized(this) {
INSTANCE ?: Singleton(context).also {
INSTANCE = it
}
}
}
}
```
### SharedPreferences
[SharedPreferences Docs](https://developer.android.com/training/data-storage/shared-preferences)
**Get a handle to shared preferences**:
- `getSharedPreferences()` — Use this if you need multiple shared preference files identified by name, which you specify with the first parameter. You can call this from any Context in your app.
- `getPreferences()` — Use this from an Activity if you need to use only one shared preference file for the activity. Because this retrieves a default shared preference file that belongs to the activity, you don't need to supply a name.
```kotlin
val sharedPref = activity?.getSharedPreferences( getString(R.string.preference_file_key), Context.MODE_PRIVATE )
val sharedPref = activity?.getPreferences(Context.MODE_PRIVATE)
```
**Write to shared preferences**:
To write to a shared preferences file, create a `SharedPreferences.Editor` by calling `edit()` on your SharedPreferences.
Pass the keys and values to write with methods such as `putInt()` and `putString()`. Then call `apply()` or `commit()` to save the changes.
```kotlin
val sharedPref = activity?.getPreferences(Context.MODE_PRIVATE) ?: return
with (sharedPref.edit()) {
putInt(getString(R.string.key), value)
commit() // or apply()
}
```
`apply()` changes the in-memory `SharedPreferences` object immediately but writes the updates to disk *asynchronously*.
Alternatively, use `commit()` to write the data to disk *synchronously*. But because *commit()* is synchronous, avoid calling it from your main thread because it could pause the UI rendering.
**Read from shared preferences**:
To retrieve values from a shared preferences file, call methods such as getInt() and getString(), providing the key for the wanted value, and optionally a default value to return if the key isn't present.
```kotlin
val sharedPref = activity?.getPreferences(Context.MODE_PRIVATE) ?: return
val defaultValue = resources.getInteger(R.integer.default_value_key)
val value = sharedPref.getInt(getString(R.string.key), defaultValue)
```

View file

@ -1,86 +0,0 @@
# RecycleView Cell Adapter
[Adapters][1] provide a binding from an app-specific data set to views that are displayed within a `RecyclerView`.
[1]:https://developer.android.com/reference/kotlin/androidx/recyclerview/widget/RecyclerView.Adapter
```kotlin
package <package>
import android.content.Context
import android.util.Log
import android.view.LayoutInflater
import android.view.ViewGroup
import android.widget.Button
import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
class RecipeAdapter : RecyclerView.Adapter<RecipeAdapter.ViewHolder> {
private var mDataset: ArrayList<Recipe> // RecyclerView data
class ViewHolder : RecyclerView.ViewHolder {
var viewGroup: ViewGroup? = null
constructor(v: ViewGroup?) : super(v!!) {
viewGroup = v
}
}
//adapter contractor, takes list of data
constructor(myDataset: ArrayList<Recipe>/*, mContext: Context?*/){
mDataset = myDataset
//mContext = mContext
}
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder {
val v : ViewGroup = LayoutInflater.from(parent.context).inflate(R.layout.view_recipe_item, parent, false) as ViewGroup
val vh = ViewHolder(v)
return vh
}
override fun getItemCount(): Int {
return mDataset.size
}
override fun onBindViewHolder(holder: ViewHolder, position: Int) {
val mItem : Recipe = mDataset.get(position)
val titleText = holder.viewGroup?.findViewById<TextView>(R.id.titleText)
titleText?.text = mItem.title
Log.d("Adapter","Title: "+mItem.title)
val deleteButton = holder.viewGroup!!.findViewById<Button>(R.id.deleteButton)
deleteButton.setOnClickListener { removeItem(position) }
//Click
holder.viewGroup?.setOnClickListener {
mListener?.select(position)
}
}
private fun removeItem(position: Int) {
mDataset.removeAt(position)
notifyItemRemoved(position)
notifyItemRangeChanged(position, mDataset.size)
}
/*
* Callback
* */
private var mListener: AdapterCallback? = null
interface AdapterCallback {
fun select(position: Int)
}
fun setOnCallback(mItemClickListener: AdapterCallback) {
this.mListener = mItemClickListener
}
}
```

View file

@ -1,154 +0,0 @@
# Fragments & FragmentActivity
[Documentation](https://developer.android.com/guide/components/fragments)
A **Fragment** represents a behavior or a portion of user interface in a `FragmentActivity`. It's possible to combine multiple fragments in a single activity to build a multi-pane UI and reuse a fragment in multiple activities.
Think of a fragment as a *modular section of an activity*, which has its *own* lifecycle, receives its *own* input events, and which you can add or remove while the activity is running (sort of like a "sub activity" that you can reuse in different activities).
![Fragment](../.images/android_fragments.png)
A fragment must always be hosted in an activity and the fragment's lifecycle is *directly affected* by the host activity's lifecycle.
![Fragment Lifecycle](../.images/android_fragment-lifecycle.png)
## Minimal Fragment Functions
Usually, you should implement at least the following lifecycle methods:
`onCreate()`
The system calls this when creating the fragment. Within your implementation, you should initialize essential components of the fragment that you want to retain when the fragment is paused or stopped, then resumed.
`onCreateView()`
The system calls this when it's time for the fragment to draw its user interface for the first time. To draw a UI for your fragment, you must return a View from this method that is the root of your fragment's layout. You can return null if the fragment does not provide a UI.
`onPause()`
The system calls this method as the first indication that the user is leaving the fragment (though it doesn't always mean the fragment is being destroyed). This is usually where you should commit any changes that should be persisted beyond the current user session (because the user might not come back).
## Fragment Subclasses
`DialogFragment`
Displays a floating dialog. Using this class to create a dialog is a good alternative to using the dialog helper methods in the Activity class, because you can incorporate a fragment dialog into the back stack of fragments managed by the activity, allowing the user to return to a dismissed fragment.
`ListFragment`
Displays a list of items that are managed by an adapter (such as a SimpleCursorAdapter), similar to ListActivity. It provides several methods for managing a list view, such as the onListItemClick() callback to handle click events. (Note that the preferred method for displaying a list is to use RecyclerView instead of ListView. In this case you would need to create a fragment that includes a RecyclerView in its layout. See Create a List with RecyclerView to learn how.)
`PreferenceFragmentCompat`
Displays a hierarchy of Preference objects as a list. This is used to create a settings screen for your application.
## Fragment Insertion in Layout (Method 1)
In `Activity.xml`:
```xml
<!-- Activity.xml boilerplate -->
<!-- Fragment drawn at Activity start cannot be drawn on event -->
<fragment
android:name="com.<app>.<Fragment>"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/fragment_id" />
<!-- Activity.xml boilerplate -->
```
In `Fragment.kt`:
```kotlin
package <package>
import android.os.Bundle
import androidx.fragment.app.Fragment
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
/**
* A simple [Fragment] subclass.
*/
class FirstFragment : Fragment() {
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? {
// Inflate the layout for this fragment (draw and insert fragment in Activity)
return inflater.inflate(R.layout.<fragment_xml>, container, false)
}
//called after fragment is drawn (ACTIONS GO HERE)
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
}
}
```
## Fragment Insertion in Layout (Method 2)
```xml
<!-- Activity.xml boilerplate -->
<layout
android:id="@+id/containerID"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<!-- Activity.xml boilerplate -->
```
In `Activity.kt`:
```kotlin
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.<activity_xml>)
val fragmentID = FragmentClass()
//insert fragment from kotlin (can be triggered by event)
supportFragmentManager
.beginTransaction()
.add(R.id.<containerID>, fragmentID)
.commit()
//replace fragment from kotlin (can be triggered by event)
supportFragmentManager
.beginTransaction()
.replace(R.id.<containerID>, fragmentID)
.addToBackStack(null) // remember order of loaded fragment (keep alive previous fragments)
.commit()
}
```
## Passing Values from Activity to Fragment
In `Activity.kt`:
```kotlin
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.<activity_xml>)
val fragmentID = FragmentClass()
val bundle = Bundle()
bundle.putString("key", "value") //passed value
fragmentID.arguments = bundle
//insert fragment from kotlin (can be triggered by event)
supportFragmentManager
.beginTransaction()
.add(R.id.<containerID>, fragmentID)
.commit()
}
```
In `Fragment.kt`:
```kotlin
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? {
value = arguments?.getString("key") //read passed value
// Inflate the layout for this fragment (draw and insert fragment in Activity)
return inflater.inflate(R.layout.<fragment_xml>, container, false)
}
```
![f](https://blog.avenuecode.com/hs-fs/hubfs/The%20activity%20and%20fragment%20lifecycles%20(1).png?width=640&name=The%20activity%20and%20fragment%20lifecycles%20(1).png)

View file

@ -1,35 +0,0 @@
# Gradle & Building
## `build.gradle` (Module)
```gradle
android {
defaultconfig{
applicationId <string> // app package name
minSdkVersion <int> // min API version
targetSdkVersion <int> // actual API version
versionCode <int> // unique code for each release
versionName "<string>"
}
}
```
## Distributing the app
### APK vs Bundle
- APK: android executable, contains all assets of the app (multiple versions for each device). The user downloads all the assets.
- Bundle: split the assets based on device specs. The users downloads only the assets needed for the target device.
### APK generation & [App Signing](https://developer.android.com/studio/publish/app-signing)
- **Key store path**: location where the keystore file will be stored.
- **Key store password**: secure password for the keystore (**to be remembered**).
- **Key alias**: identifying name for the key (**to be remembered**).
- **Key password**: secure password for the key (**to be remembered**).
The keystore identifies the app and it's developers. It's needed for APK generation and releases distribution.

View file

@ -1,84 +0,0 @@
# Maps Activity
[Google Maps Docs](https://developers.google.com/maps/documentation/android-sdk/intro)
Activity should be **Google Maps Activity**.
In `google_maps_api.xml`:
```xml
<resources>
<string name="google_maps_key" templateMergeStrategy="preserve", translatable="false">API_KEY</string>
</resources>
```
## Activity Layout (xml)
```xml
<!-- a fragment to contain the map -->
<fragment xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:map="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/<fragment_id>"
android:name="com.google.android.gms.maps.SupportMapFragment"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".<Activity>" />
```
## Activity Class (kotlin)
```kotlin
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import com.google.android.gms.maps.CameraUpdateFactory
import com.google.android.gms.maps.GoogleMap
import com.google.android.gms.maps.OnMapReadyCallback
import com.google.android.gms.maps.SupportMapFragment
import com.google.android.gms.maps.model.LatLng
import com.google.android.gms.maps.model.Marker
import com.google.android.gms.maps.model.MarkerOptions
class MapsActivity : AppCompatActivity(), OnMapReadyCallback,
GoogleMap.OnInfoWindowClickListener,
GoogleMap.OnMarkerClickListener,
GoogleMap.OnCameraIdleListener{
private lateinit var mMap: GoogleMap // declare but not valorize
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.<activity_xml>)
// add the map to a fragment
val mapFragment = supportFragmentManager.findFragmentById(R.id.<fragment_id>) as SupportMapFragment
mapFragment.getMapAsync(this)
}
override fun onMapReady(googleMap: GoogleMap) {
mMap = googleMap
mMap.setOnMarkerClickListener(this)
mMap.setOnInfoWindowClickListener(this)
// Add a marker and move the camera
val location = LatLng(-34.0, 151.0) // set location with latitude and longitude
mMap.addMarker(MarkerOptions().position(location).title("Marker in ...")) // ad the marker to the map with a name and position
mMap.moveCamera(CameraUpdateFactory.newLatLng(location)) // move camera to the marker
}
override fun onInfoWindowClick(p0: Marker?) {
}
override fun onMarkerClick(p0: Marker?): Boolean {
}
override fun onCameraIdle() {
}
}
```

View file

@ -1,101 +0,0 @@
# SQLite in Android
[Kotlin SQLite Documentation](https://developer.android.com/training/data-storage/sqlite)
[NoSQL DB Realm Docs](https://realm.io/docs)
## Database Helper
Create database in memory and its tables.
```kotlin
class DatabaseHelper(
context: Context?,
name: String?,
factory: SQLiteDatabase.CursorFactory?,
version: Int
) : SQLiteOpenHelper(context, name, factory, version) { // super constructor?
// called on db creation if it does not exists (app installation)
override fun onCreate() {
sqLiteDatabase?.execSQL("CREATE TABLE ...") // create table
}
override fun onUpgrade(sqLiteDatabase: SQLiteDatabase?, ...) {
// table update logic
}
}
```
## AppSingleton
Make data persistent.
```kotlin
class AppSingleton constructor(context: Context) {
var database: SQLiteDatabase? = null // point to database file in memory
companion object {
@Volatile
private var INSTANCE: AppSingleton? = null
// synchronized makes sure that all instances of the singleton are actually the only existing one
fun getInstance(context: Context) = INSTANCE ?: synchronized(this) {
INSTANCE ?: Singleton(context).also {
INSTANCE = it
}
}
}
// called to create DB in device memory
fun openDatabase(context: Context?) {
var databaseHelper = DatabaseHelper(context, "AppDB_Name.sqlite", null, version) {
database = databaseHelper.writeDatabase //?
}
}
}
```
## Activity
```kotlin
override fun onCreate() {
// either creates or reads DB
AppSingleton.getInstance().openDatabase(this) // context is this
val controller = ModelController()
controller.insert(model) // insert data from object
}
```
## ModelController
Controller to handle data from the objects and interact with the Database
```kotlin
sqLiteDatabase = AppISingleton.getInstance().database // reference to the database from the singleton
contentValues = ContentValues() // dict like structure to insert data in DB
contentValues.put("DB_Column", value) // put data in the structure
// insertion query V1
sqLiteDatabase?.insert("table", null, contentValue)
//insertion query V2
sqLiteDatabase?.rawQuey("INSERT INTO ...")
// returns a Cursor()
val cursor = sqLiteDatabase?.query("table", ArrayOf(columns_to_read), "WHERE ...", "GROUP BY ...", "HAVING ...", "ORDER BY ...")
try {
if (cursor != null) {
while(cursor.moveToNext()) { // loop through cursor data
data = cursor.getString(cursor.getColumnIndex("DB_Column")) // read data from the cursor
}
}
} finally {
cursor?.close()
}
```

View file

@ -1,215 +0,0 @@
# Activity.xml
## Resources
### Colors, Style, Strings
These resources are located in `app/src/main/res/values/<resource_type>.xml`
`@color/colorName` -> access to *color definition* in `colors.xml`
`@string/stringName` -> access to *string definition* in `strings.xml` (useful for localization)
`@style/styleName` -> access to *style definition* in `styles.xml`
In `colors.xml`:
```xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<!-- app bar color -->
<color name="colorPrimary">#6200EE</color>
<!-- status bar color -->
<color name="colorPrimaryDark">#3700B3</color>
<color name="colorAccent">#03DAC5</color>
<!-- other color definitions -->
</resources>
```
In `strings.xml`:
```xml
<resources>
<string name="app_name"> AppName </string>
<!-- other strings definitions -->
</resources>
```
In `styles.xml`:
```xml
<resources>
<!-- Base application theme. -->
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Customize your theme here. -->
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
<item name="colorAccent">@color/colorAccent</item>
</style>
</resources>
```
## View & View Group
A **View** contains a specific element of the layout.
A **View Group** or **Layout** is a container of Views.
```xml
<View settings/>
<ViewGroup settings>
...
</ViewGroup>
```
`android:layout_width`, `android:layout_height`:
* fixed value (dp, sp)
* match_parent
* wrap_content
## ViewGroups
### [RelativeLayout](https://developer.android.com/reference/android/widget/RelativeLayout.LayoutParams)
The relative layout containing the view uses the value of these layout parameters to determine where to position the view on the screen.
If the view is not contained within a relative layout, these attributes are ignored.
```xml
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
</RelativeLayout>
```
`android:layout_above`: Positions the bottom edge of this view above the given anchor view ID.
`android:layout_alignBaseline`: Positions the baseline of this view on the baseline of the given anchor view ID.
`android:layout_alignBottom`: Makes the bottom edge of this view match the bottom edge of the given anchor view ID.
`android:layout_alignEnd`: Makes the end edge of this view match the end edge of the given anchor view ID.
`android:layout_alignLeft`: Makes the left edge of this view match the left edge of the given anchor view ID.
`android:layout_alignParentBottom`: If true, makes the bottom edge of this view match the bottom edge of the parent.
`android:layout_alignParentEnd`: If true, makes the end edge of this view match the end edge of the parent.
`android:layout_alignParentLeft`: If true, makes the left edge of this view match the left edge of the parent.
`android:layout_alignParentRight`: If true, makes the right edge of this view match the right edge of the parent.
`android:layout_alignParentStart`: If true, makes the start edge of this view match the start edge of the parent.
`android:layout_alignParentTop`: If true, makes the top edge of this view match the top edge of the parent.
`android:layout_alignRight`: Makes the right edge of this view match the right edge of the given anchor view ID.
`android:layout_alignStart`: Makes the start edge of this view match the start edge of the given anchor view ID.
`android:layout_alignTop`: Makes the top edge of this view match the top edge of the given anchor view ID.
`android:layout_alignWithParentIfMissing`: If set to true, the parent will be used as the anchor when the anchor cannot be be found for layout_toLeftOf, layout_toRightOf, etc.
`android:layout_below`: Positions the top edge of this view below the given anchor view ID.
`android:layout_centerHorizontal`: If true, centers this child horizontally within its parent.
`android:layout_centerInParent`: If true, centers this child horizontally and vertically within its parent.
`android:layout_centerVertical`: If true, centers this child vertically within its parent.
`android:layout_toEndOf`: Positions the start edge of this view to the end of the given anchor view ID.
`android:layout_toLeftOf`: Positions the right edge of this view to the left of the given anchor view ID.
`android:layout_toRightOf`: Positions the left edge of this view to the right of the given anchor view ID.
`android:layout_toStartOf`: Positions the end edge of this view to the start of the given anchor view ID.
### [LinearLayout](https://developer.android.com/reference/android/widget/LinearLayout.LayoutParams)
Layout that arranges other views either horizontally in a single column or vertically in a single row.
```xml
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"> <!-- or horizontal -->
</LinearLayout>
```
`android:layout_weight`: Indicates how much of the extra space in the LinearLayout is allocated to the view associated with these LayoutParams.
`android:layout_margin`: Specifies extra space on the left, top, right and bottom sides of this view.
`android:layout_marginBottom`: Specifies extra space on the bottom side of this view.
`android:layout_marginEnd`: Specifies extra space on the end side of this view.
`android:layout_marginHorizontal`: Specifies extra space on the left and right sides of this view.
`android:layout_marginLeft`: Specifies extra space on the left side of this view.
`android:layout_marginRight`: Specifies extra space on the right side of this view.
`android:layout_marginStart`: Specifies extra space on the start side of this view.
`android:layout_marginTop`: Specifies extra space on the top side of this view.
`android:layout_marginVertical`: Specifies extra space on the top and bottom sides of this view.
`android:layout_height`: Specifies the basic height of the view.
`android:layout_width`: Specifies the basic width of the view.
## Views
```xml
<!--ID is necessary for identification -->
<View
android:id="@+id/uniqueId"
android:layout_width="value"
android:layout_height="value"
/>
```
### TextView
To add `...` to truncate a string:
```xml
<!-- maxLines + ellipsize works only in API 22+ -->
<TextView
maxLines="n"
ellipsize="end" />
```
### ScrollView
The scroll view can only have one child. If the child is a layout than the layout can contain many elements.
```xml
<ScrollView
android:id="@+id/uniqueId"
android:layout_width="value"
android:layout_height="value">
<!-- single child -->
</ScrollView>
```
### RecyclerView
List of items
In `build.gradle`:
```gradle
dependencies {
implementation "androidx.recyclerview:recyclerview:<version>"
// For control over item selection of both touch and mouse driven selection
implementation "androidx.recyclerview:recyclerview-selection:<version>"
}
```
In `activity.xml`:
```xml
<!-- VIewGroup for a list of items -->
<androidx.recycleview.widget.RecycleView
android:id="@+id/uniquieId"
android:layout_width="value"
android:layout_height="value">
```
In `recyclerViewItem.xml`:
```xml
<!-- list item layout -->
```
### WebView
```xml
<WebView
android:id="@+id/uniquieId"
android:layout_width="value"
android:layout_height="value">
```
## Constraints
If a view is *anchored on both sides* and the width/height is set to `0dp` the view will expand to touch the elements to which is anchored.

View file

@ -1,14 +0,0 @@
# App.swift
```swift
import SwiftUI
@main
struct <Project>App: App {
var body: some Scene {
WindowGroup {
ContentView()
}
}
}
```

View file

@ -1,75 +0,0 @@
# AppDelegate
First loaded file. It prepares the app visualisation by calling the *scene* UI.
```swift
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
// Override point for customization after application launch.
return true
}
// MARK: UISceneSession Lifecycle
func application(_ application: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options: UIScene.ConnectionOptions) -> UISceneConfiguration {
// Called when a new scene session is being created.
// Use this method to select a configuration to create the new scene with.
return UISceneConfiguration(name: "Default Configuration", sessionRole: connectingSceneSession.role)
}
func application(_ application: UIApplication, didDiscardSceneSessions sceneSessions: Set<UISceneSession>) {
// Called when the user discards a scene session.
// If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions.
// Use this method to release any resources that were specific to the discarded scenes, as they will not return.
}
// MARK: - Core Data stack
lazy var persistentContainer: NSPersistentContainer = {
/*
The persistent container for the application. This implementation
creates and returns a container, having loaded the store for the
application to it. This property is optional since there are legitimate
error conditions that could cause the creation of the store to fail.
*/
let container = NSPersistentContainer(name: "CoreDataExample")
container.loadPersistentStores(completionHandler: { (storeDescription, error) in
if let error = error as NSError? {
// Replace this implementation with code to handle the error appropriately.
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
/*
Typical reasons for an error here include:
* The parent directory does not exist, cannot be created, or disallows writing.
* The persistent store is not accessible, due to permissions or data protection when the device is locked.
* The device is out of space.
* The store could not be migrated to the current model version.
Check the error message to determine what the actual problem was.
*/
fatalError("Unresolved error \(error), \(error.userInfo)")
}
})
return container
}()
// MARK: - Core Data Saving support
func saveContext () {
let context = persistentContainer.viewContext
if context.hasChanges {
do {
try context.save()
} catch {
// Replace this implementation with code to handle the error appropriately.
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
let nserror = error as NSError
fatalError("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
```

View file

@ -1,58 +0,0 @@
# SceneDelegate
Loads the *scene* and set the `rootController` to control the app. The `ContentView` gets called, which contains the UI elements to be visualized.
```swift
import UIKit
import SwiftUI
class SceneDelegate: UIResponder, UIWindowSceneDelegate {
var window: UIWindow?
func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
// Use this method to optionally configure and attach the UIWindow `window` to the provided UIWindowScene `scene`.
// If using a storyboard, the `window` property will automatically be initialized and attached to the scene.
// This delegate does not imply the connecting scene or session are new (see `application:configurationForConnectingSceneSession` instead).
// Create the SwiftUI view that provides the window contents.
let contentView = ContentView()
// Use a UIHostingController as window root view controller.
if let windowScene = scene as? UIWindowScene {
let window = UIWindow(windowScene: windowScene)
window.rootViewController = UIHostingController(rootView: contentView)
self.window = window
window.makeKeyAndVisible()
}
}
func sceneDidDisconnect(_ scene: UIScene) {
// Called as the scene is being released by the system.
// This occurs shortly after the scene enters the background, or when its session is discarded.
// Release any resources associated with this scene that can be re-created the next time the scene connects.
// The scene may re-connect later, as its session was not necessarily discarded (see `application:didDiscardSceneSessions` instead).
}
func sceneDidBecomeActive(_ scene: UIScene) {
// Called when the scene has moved from an inactive state to an active state.
// Use this method to restart any tasks that were paused (or not yet started) when the scene was inactive.
}
func sceneWillResignActive(_ scene: UIScene) {
// Called when the scene will move from an active state to an inactive state.
// This may occur due to temporary interruptions (ex. an incoming phone call).
}
func sceneWillEnterForeground(_ scene: UIScene) {
// Called as the scene transitions from the background to the foreground.
// Use this method to undo the changes made on entering the background.
}
func sceneDidEnterBackground(_ scene: UIScene) {
// Called as the scene transitions from the foreground to the background.
// Use this method to save data, release shared resources, and store enough scene-specific state information
// to restore the scene back to its current state.
}
}
```

Some files were not shown because too many files have changed in this diff Show more