해결방법 

1. 구글 플레이 스토어 실행

2. 상단 왼쪽에 메뉴아이콘 클릭, play 프로트 선택

3.기기에 보안 위협이 웅앵웅 체크 해제

 

참조 : https://m.blog.naver.com/jaseazer/221317744472

 

android6.0 이하 버전에서는 글쓰기 작성 완료 버튼을 눌러도 그 후 작동을 하지 않는 에러 발생.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
        @SuppressWarnings("deprecation")
        write_web_view.webViewClient = object : WebViewClient() {
            @SuppressWarnings("deprecation")
            override fun onPageStarted(view: WebView?, url: String?, favicon: Bitmap?) {
                if (url.contains("toapp")) {
                }
            }
 
            override fun shouldOverrideUrlLoading(webView: WebView?, url: String?): Boolean {
 
                if (url.contains("toapp")) {
                    //       didOccureContentDetail(url)
                } else {
                    webView.loadUrl(url)
                }
                return true
            }
 
            @TargetApi(Build.VERSION_CODES.N)
            override fun shouldOverrideUrlLoading(view: WebView?, request: WebResourceRequest?): Boolean {
 
                if (request!!.url.toString().contains("toapp")) {
                    didOccureContentDetail(request.url.toString())
                } else {
                    write_web_view.loadUrl(request.url.toString())
                }
                return true
            }
        }
cs

 

 이유는 모르겠지만 android version 6.0(Mashmellow) 이하 버전에서 반응하는 메소드의 내부 구현이 주석처리 되어있었다.^^..ㅋ...

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
        @SuppressWarnings("deprecation")
        write_web_view.webViewClient = object : WebViewClient() {
            @SuppressWarnings("deprecation")
            override fun onPageStarted(view: WebView?, url: String?, favicon: Bitmap?) {
                if (url!!.contains("toapp")) {
                }
            }
 
            override fun shouldOverrideUrlLoading(webView: WebView?, url: String?): Boolean {
 
                if (url!!.contains("toapp")) {
                           didOccureContentDetail(url)
                } else {
                    webView!!.loadUrl(url)
                }
                return true
            }
 
            @TargetApi(Build.VERSION_CODES.N)
            override fun shouldOverrideUrlLoading(view: WebView?, request: WebResourceRequest?): Boolean {
 
                if (request!!.url.toString().contains("toapp")) {
                    didOccureContentDetail(request.url.toString())
                } else {
                    write_web_view.loadUrl(request.url.toString())
                }
                return true
            }
        }
cs

 

null값을 받을 수 없는데 null 값이 들어왔기 때문에 발생한 에러.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
        @SuppressWarnings("deprecation")
        write_web_view.webViewClient = object : WebViewClient() {
            @SuppressWarnings("deprecation")
            override fun onPageStarted(view: WebView, url: String, favicon: Bitmap) {
                if (url.contains("toapp")) {
                }
            }
 
            override fun shouldOverrideUrlLoading(webView: WebView, url: String): Boolean {
 
                if (url.contains("toapp")) {
                    //       didOccureContentDetail(url)
                } else {
                    webView.loadUrl(url)
                }
                return true
            }
 
            @TargetApi(Build.VERSION_CODES.N)
            override fun shouldOverrideUrlLoading(view: WebView?, request: WebResourceRequest?): Boolean {
 
                if (request!!.url.toString().contains("toapp")) {
                    didOccureContentDetail(request.url.toString())
                } else {
                    write_web_view.loadUrl(request.url.toString())
                }
                return true
            }
        }
cs

 

Nullable로 수정해서 구현 완료.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
        @SuppressWarnings("deprecation")
        write_web_view.webViewClient = object : WebViewClient() {
            @SuppressWarnings("deprecation")
            override fun onPageStarted(view: WebView?, url: String?, favicon: Bitmap?) {
                if (url!!.contains("toapp")) {
                }
            }
 
            override fun shouldOverrideUrlLoading(webView: WebView?, url: String?): Boolean {
 
                if (url!!.contains("toapp")) {
                    //       didOccureContentDetail(url)
                } else {
                    webView!!.loadUrl(url)
                }
                return true
            }
 
            @TargetApi(Build.VERSION_CODES.N)
            override fun shouldOverrideUrlLoading(view: WebView?, request: WebResourceRequest?): Boolean {
 
                if (request!!.url.toString().contains("toapp")) {
                    didOccureContentDetail(request.url.toString())
                } else {
                    write_web_view.loadUrl(request.url.toString())
                }
                return true
            }
        }
cs

 

xzing 라이브러리를 이용한 QR코드 구현중 발생한 에러.

error 내용

1
2
3
    java.lang.RuntimeException: Failure delivering result ResultInfo{who=null, request=49374, result=0, data=null} to activity 
{com.example.qrcodetest/com.example.qrcodetest.ActivityMain}: java.lang.IllegalArgumentException: 
Parameter specified as non-null is null: method kotlin.jvm.internal.Intrinsics.checkParameterIsNotNull, parameter intent

 

1
2
3
4
5
6
    override fun onActivityResult(requestCode: Int, resultCode: Int, intent: Intent) {
        if (resultCode == Activity.RESULT_OK) {
            val scanResult:IntentResult = IntentIntegrator.parseActivityResult(requestCode, resultCode, intent)
           // some code
        }
    }
cs
cs

뒤로가기 버튼을 눌렀을 때 this.finish() 를 호출하여 현재 액티비티를 종료하게 만들었는데

맨 처음 activity-> activity 구현시엔 아무런 오류가 없었던 게

activity->view로 바꾸니 갑자기 RuntimeException, IllegalArgumentException를 보냈다.

해당 액티비티가 종료되면 onActivityResult 메소드가 무조건 호출되는 방식인데

QR코드를 찍지 않고 뒤로가기를 눌러 종료하였을 때 data가 null값인게 문제가 되었던것이다.

그래서 데이터를 받아오는 intent 변수에 nullable 기호인 ?를 붙여줬더니 정상작동 하였다.

(왜 activity->activity 였을땐 아무런 문제가 없었는지는 모름 ㅠ)

1
2
3
4
5
6
7
  override fun onActivityResult(requestCode: Int, resultCode: Int, intent: Intent?) {
        if (resultCode == Activity.RESULT_OK) {
            val scanResult:IntentResult = IntentIntegrator.parseActivityResult(requestCode, resultCode, intent)
            //some code
            }
        }
    }
cs
  • 코틀린은 자바랑 거의 비슷하지만, 변수선언에서는 약간의 차이가 있다.

  • 코틀린의 변수선언은 변수의 타입을 꼭 선언하지 않아도 된다는 특징이 있다. (단, 생성자의 매개변수로 들어갈 때는 명시해아함.)

  • 변수 선언 방법 
1
2
3
4
5
6
7
8
9
10
11
  // val/var 변수명 : 변수타입 = 초기화
 
 
    // Boolean 타입 
    private var clickedButton: Boolean = false
 
    // String 타입(자동 유추)
    protected val mUserKey = "user1234"
 
    // int 타입
    public var getNum: Int? = null
cs

 

val/var

  • val : valuable의 약자, 읽기만 가능한 final 변수.

  • var : variable의 약자, 읽기/쓰기가 가능한 일반 변수.

 

Nullable/Non-null

  • 코틀린은 null값을 가질 수 있는 변수를  '타입?' 로 선언한다. 이 변수는 null check가 필요하다.

  •   '변수명!!' 로 해당 변수가 Null이 아니라고 명시할 수 있다.

  • 당연한 얘기이지만 Non-null로 선언된 변수는 Nullable 변수를 받아올 수 없다. 

  • 따라서 Non-null로 선언된 변수에 Nullable 변수를 받아오기 위해선 다음과 같이 선언하면 된다.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
  @NonNull
    var name: String = "yoon"
 
    @Nullable
    fun GetMyName(name: String):String?{
        return name
    }
 
    // 에러가 뜬다.
     @NonNull
    val mmMyname: String = GetMyName(name)
 
    // 에러가 뜨지않는다.
     @NonNull
    val mmMyName2: String = (GetMyName(name))!!
cs

 

 

참조 : https://brunch.co.kr/@mystoryg/8

'개발 > Kotlin' 카테고리의 다른 글

[Kotlin] Companion object  (0) 2019.04.20
  • 보통 새로운 액티비티는 오른쪽->왼쪽 방향으로 들어오고 끝나는 액티비티는 왼쪽->오른쪽 방향으로 나간다.

  • 구현상 문제로 강제로 설정해야 할 때 사용하면 된다.

 

- res/anim 폴더 안에 생성 (xml)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
// anim_slide_in_left.xml (오른쪽->왼쪽 방향으로  들어오는 애니메이션)
 
<translate xmlns:android="http://schemas.android.com/apk/res/android"
 
    android:duration="250"
 
    android:fromXDelta="-100%"
 
    android:interpolator="@android:anim/decelerate_interpolator"
 
    android:toXDelta="0%" />
 
 
 
 
 
// anim_slide_out_right.xml (왼쪽->오른쪽 방향으로 나가는 애니메이션)
 
<translate xmlns:android="http://schemas.android.com/apk/res/android"
 
android:duration="250"
 
android:fromXDelta="0%"
 
android:interpolator="@android:anim/decelerate_interpolator"
 
android:toXDelta="100%" />
cs

 

- class 내에 선언 방법 (java)

1
2
3
startActivity(new Intent(현재Activity.this, 불러올Activity.class));
 
overridePendingTransition(R.anim.새로운(들어오는)Activity애니메이션R.anim.현재(사라질)Activity애니메이션);
cs

 

- 구현 예제 (kotlin)

1
2
3
4
5
6
7
8
9
10
11
fun backButtonClick(v: View?) {
 
this.finish()
 
val intent = Intent(this, ActivityQNAList::class.java)
 
startActivity(intent)
 
overridePendingTransition(R.anim.anim_slide_in_left, R.anim.anim_slide_out_right)
 
}
cs

 

 

 

참조: https://dwfox.tistory.com/26 [DWFOX]

  • mipmap은 보통 사용하지않는다. (drawable 이용)

  • drawable 이용시 drawable-xxxhdpi 추가

 

'개발 > 메모' 카테고리의 다른 글

[Git] 잔디가 안심어질때(for mac)  (0) 2021.08.04

코틀린은 static이 존재하지 않는다. 따라서 static을 사용하기 위해선 object로 선언해야한다.

 

1. 일반 객체 선언

  • 일반 객체 선언은 보통 싱글톤으로 이용이 가능하다.
  • 클래스 선언과 그 클래스에 속한 단일 인스턴스 생성을 동시에 처리한다.
  • 보통의 클래스처럼 다른 클래스와 인터페이스의 상속이 가능하다.
  • property, method, init 선언이 가능하다.

 

- companion object

  • 코틀린은 한 클래스 내에 inner와 object를 동시에 구현할 수 없다. 클래스 안에 들어있는 객체 선언일지라도 바깥 클래스 멤버에 그냥 접근할 수 없다.

  • 하지만 객체를 생성하면서 접근하는 방법은 가능하다. private 생성자를 object 내에서 호출할 수 있어 팩토리 메소드 패턴을 구현하기 적합하다.

  • 클래스 안에 중첩 클래스로 object가 들어가는 경우 앞에 companion을 붙이면 object의 이름을 명시하지 않고 바깥 클래스의 이름으로 바로 접근할 수 있다.

  • 호출 : 클래스.메소드이름()

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
// Companion object가 선언된 팝업 클래스
class _Popup {
    companion object {
        private val mInstance = _Popup()
        fun GetInstance(): _Popup {
            return mInstance
        }
    }
           fun ShowConfirmPopup(context: Context, title: String?, mainMessage: String?, confirmMessage: String) {
        mConfirmAlt_Builder = AlertDialog.Builder(context)
        if (title != null) mConfirmAlt_Builder.setTitle(title)
        if (mainMessage != null) mConfirmAlt_Builder.setMessage(mainMessage)
        mConfirmAlt_Builder.setCancelable(false)
        mConfirmAlt_Builder.setPositiveButton(confirmMessage) { dialog, id ->
            if (confirmPopupListener != null) confirmPopupListener.didSelectConfirmPopup(title, mainMessage, confirmMessage)
            binaryPopupListener = nul
            mBinaryAlt_Builder = null
            dialog.cancel()
        }
        val mmAlert = mConfirmAlt_Builder.create()
        mmAlert.show()
    }
}
// 버튼을 누르면 팝업을 불러오는 액티비티 클래스
class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        temp_button.setOnClickListener {
            _Popup.GetInstance().ShowBinaryPopup(
                    this,
                    "앱을 종료하시겠습니까?",
                    "종료",
                    "취소",
                    object : _Popup.BinaryPopupListener {
                        override fun didSelectBinaryPopup(mainMessage: String, selectMessage: String) {
                            if (selectMessage == "종료") {
                                exitApp()
                            } else {
                                return
                            }
                        }
                    })
        }
    }
    // 앱 종료하는 메소드
    private fun exitApp() {
        this.finish()
    }
}
cs

 

 

  • 결과 화면

 

참조 : https://umbum.tistory.com/597

'개발 > Kotlin' 카테고리의 다른 글

[Kotlin] val, var  (0) 2019.04.21

+ Recent posts