1 /* 2 * Copyright 2019 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.egg.quares 18 19 import android.app.Activity 20 import android.content.Context 21 import android.content.res.Configuration 22 import android.graphics.Canvas 23 import android.graphics.Paint 24 import android.graphics.Typeface 25 import android.graphics.drawable.Icon 26 import android.os.Bundle 27 import android.text.StaticLayout 28 import android.text.TextPaint 29 import android.util.Log 30 import android.view.View 31 import android.view.View.GONE 32 import android.view.View.VISIBLE 33 import android.widget.Button 34 import android.widget.CompoundButton 35 import android.widget.GridLayout 36 37 import java.util.Random 38 39 import com.android.egg.R 40 41 const val TAG = "Quares" 42 43 class QuaresActivity : Activity() { 44 private var q: Quare = Quare(16, 16, 1) 45 private var resId = 0 46 private var resName = "" 47 private var icon: Icon? = null 48 49 private lateinit var label: Button 50 private lateinit var grid: GridLayout 51 52 @Suppress("DEPRECATION") onCreatenull53 override fun onCreate(savedInstanceState: Bundle?) { 54 super.onCreate(savedInstanceState) 55 56 window.decorView.systemUiVisibility = 57 View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION or View.SYSTEM_UI_FLAG_LAYOUT_STABLE 58 59 actionBar?.hide() 60 61 setContentView(R.layout.activity_quares) 62 63 grid = requireViewById(R.id.grid) 64 label = requireViewById(R.id.label) 65 66 if (savedInstanceState != null) { 67 Log.v(TAG, "restoring puzzle from state") 68 q = savedInstanceState.getParcelable("q") ?: q 69 resId = savedInstanceState.getInt("resId") 70 resName = savedInstanceState.getString("resName", "") 71 loadPuzzle() 72 } 73 74 label.setOnClickListener { newPuzzle() } 75 } 76 onResumenull77 override fun onResume() { 78 super.onResume() 79 if (resId == 0) { 80 // lazy init from onCreate 81 newPuzzle() 82 } 83 checkVictory() 84 } 85 onSaveInstanceStatenull86 override fun onSaveInstanceState(outState: Bundle) { 87 super.onSaveInstanceState(outState) 88 89 outState.putParcelable("q", q) 90 outState.putInt("resId", resId) 91 outState.putString("resName", resName) 92 } 93 newPuzzlenull94 fun newPuzzle() { 95 Log.v(TAG, "new puzzle...") 96 97 q.resetUserMarks() 98 val oldResId = resId 99 resId = android.R.drawable.stat_sys_warning 100 try { 101 for (tries in 0..3) { 102 val ar = resources.obtainTypedArray(R.array.puzzles) 103 val newName = ar.getString(Random().nextInt(ar.length())) 104 if (newName == null) continue 105 106 Log.v(TAG, "Looking for icon " + newName) 107 108 val pkg = getPackageNameForResourceName(newName) 109 val newId = packageManager.getResourcesForApplication(pkg) 110 .getIdentifier(newName, "drawable", pkg) 111 if (newId == 0) { 112 Log.v(TAG, "oops, " + newName + " doesn't resolve from pkg " + pkg) 113 } else if (newId != oldResId) { 114 // got a good one 115 resId = newId 116 resName = newName 117 break 118 } 119 } 120 } catch (e: RuntimeException) { 121 Log.v(TAG, "problem loading puzzle, using fallback", e) 122 } 123 loadPuzzle() 124 } 125 getPackageNameForResourceNamenull126 fun getPackageNameForResourceName(name: String): String { 127 return if (name.contains(":") && !name.startsWith("android:")) { 128 name.substring(0, name.indexOf(":")) 129 } else { 130 packageName 131 } 132 } 133 checkVictorynull134 fun checkVictory() { 135 if (q.check()) { 136 val dp = resources.displayMetrics.density 137 138 val label: Button = requireViewById(R.id.label) 139 label.text = resName.replace(Regex("^.*/"), "") 140 val drawable = icon?.loadDrawable(this)?.also { 141 it.setBounds(0, 0, (32 * dp).toInt(), (32 * dp).toInt()) 142 it.setTint(label.currentTextColor) 143 } 144 label.setCompoundDrawables(drawable, null, null, null) 145 146 label.visibility = VISIBLE 147 } else { 148 label.visibility = GONE 149 } 150 } 151 loadPuzzlenull152 fun loadPuzzle() { 153 Log.v(TAG, "loading " + resName + " at " + q.width + "x" + q.height) 154 155 val dp = resources.displayMetrics.density 156 157 icon = Icon.createWithResource(getPackageNameForResourceName(resName), resId) 158 q.load(this, icon!!) 159 160 if (q.isBlank()) { 161 // this is a really boring puzzle, let's try again 162 resId = 0 163 resName = "" 164 recreate() 165 return 166 } 167 168 grid.removeAllViews() 169 grid.columnCount = q.width + 1 170 grid.rowCount = q.height + 1 171 172 label.visibility = GONE 173 174 val orientation = resources.configuration.orientation 175 176 // clean this up a bit 177 val minSide = resources.configuration.smallestScreenWidthDp - 25 // ish 178 val size = (minSide / (q.height + 0.5) * dp).toInt() 179 180 val sb = StringBuffer() 181 182 for (j in 0 until grid.rowCount) { 183 for (i in 0 until grid.columnCount) { 184 val tv: View 185 val params = GridLayout.LayoutParams().also { 186 it.width = size 187 it.height = size 188 it.setMargins(1, 1, 1, 1) 189 it.rowSpec = GridLayout.spec(GridLayout.UNDEFINED, GridLayout.TOP) // UGH 190 } 191 val x = i - 1 192 val y = j - 1 193 if (i > 0 && j > 0) { 194 if (i == 1 && j > 1) sb.append("\n") 195 sb.append(if (q.getDataAt(x, y) == 0) " " else "X") 196 tv = PixelButton(this) 197 tv.isChecked = q.getUserMark(x, y) != 0 198 tv.setOnClickListener { 199 q.setUserMark(x, y, if (tv.isChecked) 0xFF else 0) 200 val columnCorrect = (grid.getChildAt(i) as? ClueView)?.check(q) ?: false 201 val rowCorrect = (grid.getChildAt(j*(grid.columnCount)) as? ClueView) 202 ?.check(q) ?: false 203 if (columnCorrect && rowCorrect) { 204 checkVictory() 205 } else { 206 label.visibility = GONE 207 } 208 } 209 } else if (i == j) { // 0,0 210 tv = View(this) 211 tv.visibility = GONE 212 } else { 213 tv = ClueView(this) 214 if (j == 0) { 215 tv.textRotation = 90f 216 if (orientation == Configuration.ORIENTATION_LANDSCAPE) { 217 params.height /= 2 218 tv.showText = false 219 } else { 220 params.height = (96 * dp).toInt() 221 } 222 if (x >= 0) { 223 tv.setColumn(q, x) 224 } 225 } 226 if (i == 0) { 227 if (orientation == Configuration.ORIENTATION_PORTRAIT) { 228 params.width /= 2 229 tv.showText = false 230 } else { 231 params.width = (96 * dp).toInt() 232 } 233 if (y >= 0) { 234 tv.setRow(q, y) 235 } 236 } 237 } 238 grid.addView(tv, params) 239 } 240 } 241 242 Log.v(TAG, "icon: \n" + sb) 243 } 244 } 245 246 class PixelButton(context: Context) : CompoundButton(context) { 247 init { 248 setBackgroundResource(R.drawable.pixel_bg) 249 isClickable = true 250 isEnabled = true 251 } 252 } 253 254 class ClueView(context: Context) : View(context) { 255 var row: Int = -1 256 var column: Int = -1 257 var textRotation: Float = 0f 258 var text: CharSequence = "" 259 var showText = true 260 val paint: TextPaint 261 val incorrectColor: Int 262 val correctColor: Int 263 264 init { 265 setBackgroundColor(0) <lambda>null266 paint = TextPaint().also { 267 it.textSize = 14f * context.resources.displayMetrics.density 268 it.color = context.getColor(R.color.q_clue_text) 269 it.typeface = Typeface.DEFAULT_BOLD 270 it.textAlign = Paint.Align.CENTER 271 } 272 incorrectColor = context.getColor(R.color.q_clue_bg) 273 correctColor = context.getColor(R.color.q_clue_bg_correct) 274 } 275 setRownull276 fun setRow(q: Quare, row: Int): Boolean { 277 this.row = row 278 this.column = -1 279 this.textRotation = 0f 280 text = q.getRowClue(row).joinToString("-") 281 return check(q) 282 } setColumnnull283 fun setColumn(q: Quare, column: Int): Boolean { 284 this.column = column 285 this.row = -1 286 this.textRotation = 90f 287 text = q.getColumnClue(column).joinToString("-") 288 return check(q) 289 } checknull290 fun check(q: Quare): Boolean { 291 val correct = q.check(column, row) 292 setBackgroundColor(if (correct) correctColor else incorrectColor) 293 return correct 294 } 295 onDrawnull296 override fun onDraw(canvas: Canvas) { 297 super.onDraw(canvas) 298 if (!showText) return 299 canvas?.let { 300 val x = canvas.width / 2f 301 val y = canvas.height / 2f 302 var textWidth = canvas.width 303 if (textRotation != 0f) { 304 canvas.rotate(textRotation, x, y) 305 textWidth = canvas.height 306 } 307 val textLayout = StaticLayout.Builder.obtain( 308 text, 0, text.length, paint, textWidth).build() 309 canvas.translate(x, y - textLayout.height / 2) 310 textLayout.draw(canvas) 311 } 312 } 313 } 314