RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / user-558892

mister_svinia's questions

Martin Hope
mister_svinia
Asked: 2023-10-09 04:13:37 +0000 UTC

在 MainUIThread 中访问小部件时出错(两种相同的情况,一种发生,另一种则不发生)

  • 4

我有两个小部件:

<TextView
    android:id="@+id/debugView"
    android:layout_width="256dp"
    android:layout_height="128dp"
    android:layout_marginBottom="120dp"
    app:layout_constraintBottom_toBottomOf="parent"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintHorizontal_bias="0.5"
    app:layout_constraintStart_toStartOf="parent" />

<TextView
    android:id="@+id/activityAuthorization_widgets_responseStatus"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    app:layout_constraintBottom_toTopOf="@+id/activityAuthorization_widgets_login"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toBottomOf="@+id/activityAuthorization_widgets_inputPassword" />

有这个活动(未完成):

class ActivityAuthorization : AppCompatActivity() {
    private val httpClient = HttpClient()

    private lateinit var debugView: TextView

    private lateinit var inputLogin: TextInputEditText
    private lateinit var inputPassword: TextInputEditText
    private lateinit var buttonLogIn: Button
    private lateinit var showPassword: CheckBox
    private lateinit var responseStatusView: TextView

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_authorization)

        supportActionBar?.hide()

        binder()
    }

    @SuppressLint("SetTextI18n")
    @OptIn(DelicateCoroutinesApi::class)
    private fun binder() {
        debugView = findViewById(R.id.debugView)

        inputLogin = findViewById(R.id.activityAuthorization_widgets_inputLogin)
        inputPassword = findViewById(R.id.activityAuthorization_widgets_inputPassword)
        buttonLogIn = findViewById(R.id.activityAuthorization_widgets_login)

        showPassword = findViewById(R.id.activityAuthorization_widgets_showPassword)
        responseStatusView = findViewById(R.id.activityAuthorization_widgets_responseStatus)

        buttonLogIn.setOnClickListener {
            GlobalScope.launch(Dispatchers.IO) {
                val response: HttpResponse =
                    httpClient.request(
                        URLS.AUTHORIZATION
                    ) {
                        method = HttpMethod.Get
                        url {
                            parameters.append(
                                URLS.AUTHORIZATION_USERNAME,
                                inputLogin.text.toString()
                            )
                            parameters.append(
                                URLS.AUTHORIZATION_PASSWORD,
                                inputPassword.text.toString()
                            )
                        }
                    }

                when (response.status.value) {
                    200 -> completeAuthorization(response.bodyAsText())
                    404 -> userNotFound()
                    else -> unprocessedResponse()
                }
            }
        }

        showPassword.setOnCheckedChangeListener { _, isChecked ->
            if (isChecked) inputPassword.setInputType(InputType.TYPE_CLASS_TEXT)
            else inputPassword.setInputType(InputType.TYPE_CLASS_TEXT or InputType.TYPE_TEXT_VARIATION_PASSWORD)
        }
    }

    @SuppressLint("SetTextI18n")
    private fun completeAuthorization(responseText: String) {
        val responseData = JsonIterator.deserialize(
            responseText,
            ResponsePackets.AC.Authorization.Done::class.java
        )

        try {
            responseData?.apply {
                debugView.text = "id: ${
                    responseData.data.id
                }\ntoken: ${responseData.data.token}"
            } ?: unprocessedResponse()
        } catch (_: NullPointerException) { unprocessedResponse() }
    }

    @SuppressLint("SetTextI18n")
    private fun userNotFound() {
        responseStatusView.text = "123"
    }

    @SuppressLint("SetTextI18n")
    private fun unprocessedResponse() {
        responseStatusView.text = "123"
    }
}

服务器(Python Django)返回三个之一(取决于结果)

return JsonResponse({
    'status': 'Done',
    'data': {
        'id': id,
        'token': helper.DBOperator.create_token(request.META.get('HTTP_USER_AGENT'), id)
    }
}, status=200)

return JsonResponse({
    'status': 'Refused',
    'reason': 'BadRequest',
    'description': 'UserNotFound'
}, status=404)

return JsonResponse({
    'status': 'Refused',
    'reason': 'BadRequest',
    'description': 'LackOfArguments'
}, status=406)

好吧,429,但我出于调试目的将其注释掉了。

如果状态200通过并且数据正确(而且确实是正确的),那么就id正常token显示。没有错误发生。但如果返回的不是 200,那么我会切换到unprocessedResponse()(暂时也是暂时的)并得到:

2023-10-08 23:03:39.732  5012-5180  AndroidRuntime          com.gct.cl.android                   E  FATAL EXCEPTION: DefaultDispatcher-worker-4
                                                                                                    Process: com.gct.cl.android, PID: 5012
                                                                                                    android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
                                                                                                        at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:11586)
                                                                                                        at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:2648)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at androidx.constraintlayout.widget.ConstraintLayout.requestLayout(ConstraintLayout.java:3605)
                                                                                                        at android.view.View.requestLayout(View.java:27623)
                                                                                                        at android.widget.TextView.checkForRelayout(TextView.java:10904)
                                                                                                        at android.widget.TextView.setText(TextView.java:6965)
                                                                                                        at android.widget.TextView.setText(TextView.java:6751)
                                                                                                        at android.widget.TextView.setText(TextView.java:6703)
                                                                                                        at com.gct.cl.android.ActivityAuthorization.userNotFound(ActivityAuthorization.kt:106)
                                                                                                        at com.gct.cl.android.ActivityAuthorization.access$userNotFound(ActivityAuthorization.kt:23)
                                                                                                        at com.gct.cl.android.ActivityAuthorization$binder$1$1.invokeSuspend(ActivityAuthorization.kt:76)
                                                                                                        at kotlin.coroutines.jvm.internal.BaseContinuationImpl.resumeWith(ContinuationImpl.kt:33)
                                                                                                        at kotlinx.coroutines.DispatchedTask.run(DispatchedTask.kt:108)
                                                                                                        at kotlinx.coroutines.internal.LimitedDispatcher$Worker.run(LimitedDispatcher.kt:115)
                                                                                                        at kotlinx.coroutines.scheduling.TaskImpl.run(Tasks.kt:103)
                                                                                                        at kotlinx.coroutines.scheduling.CoroutineScheduler.runSafely(CoroutineScheduler.kt:584)
                                                                                                        at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.executeTask(CoroutineScheduler.kt:793)
                                                                                                        at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.runWorker(CoroutineScheduler.kt:697)
                                                                                                        at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.run(CoroutineScheduler.kt:684)
                                                                                                        Suppressed: kotlinx.coroutines.internal.DiagnosticCoroutineContextException: [StandaloneCoroutine{Cancelling}@a649923, Dispatchers.IO]

据我了解,该错误是由于尝试responseStatusView从发出请求的 Coruntine 进行访问而发生的。但为什么GET 200我没有收到类似的状态错误呢?为了以防万一,我附加了模式类和存储类:

package com.gct.cl.android;


public class ResponsePackets {
    static class AC {
        static class Authorization {
            static class Done {
                String status;

                AuthData data;
            }
            static class ClientError {
                String status;
                String reason;
                String description;
            }
            static class NotFound {
                String status;
                String reason;
                String description;
            }
        }
        static class AuthData {
            long id;
            String token;
        }
    }
}
package com.gct.cl.android

object URLS {
    const val AUTHORIZATION = "http://192.168.196.60:8082/auth" // Я подключаюсь через прокси
    const val AUTHORIZATION_USERNAME = "username"
    const val AUTHORIZATION_PASSWORD = "password"
}

UDP 我将文本输出移至debugView:

@SuppressLint("SetTextI18n")
private fun completeAuthorization(responseText: String) {
    val responseData = JsonIterator.deserialize(
        responseText,
        ResponsePackets.AC.Authorization.Done::class.java
    )
    try {
        runOnUiThread {
            Thread {
                responseData?.apply {
                    debugView.text = "id: ${responseData.data.id}\ntoken: ${responseData.data.token}"
                } ?: unprocessedResponse()
            }.start()
        }
    } catch (_: NullPointerException) {
        unprocessedResponse()
    }
}

@SuppressLint("SetTextI18n")
private fun userNotFound() {
    debugView.text = "123"
}

@SuppressLint("SetTextI18n")
private fun unprocessedResponse() {
    debugView.text = "123"
}

Aaand...一切都按预期进行。

我尝试这样做:

@SuppressLint("SetTextI18n")
private fun userNotFound() {
    runOnUiThread {
        Thread {
            debugView.setText(R.string.authorization_widgets_responseStatus_userNotFound.toString())
        }
    }
}

Aaand...同样的错误。该代码在 4 部手机和两个模拟器上进行了测试。结果到处都一样。

java
  • 1 个回答
  • 45 Views
Martin Hope
mister_svinia
Asked: 2023-10-05 03:50:09 +0000 UTC

类中的无效字段“”

  • 5

这是课程:

import at.favre.lib.crypto.bcrypt.BCrypt;
import com.jsoniter.JsonIterator;
import com.jsoniter.any.Any;
import org.jetbrains.annotations.Contract;
import org.jetbrains.annotations.NotNull;

import java.nio.charset.StandardCharsets;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;

import static java.util.Map.entry;

public class Helper {
    public static class ConnectionPath {
        String[] pathParts;
        Map<String, String> params;

        public ConnectionPath (String[] pathParts, Map<String, String> params) {
            this.pathParts = pathParts;
            this.params = params;
        }
    }

    public static class MessagePacket {
        final String hash;
        final String[] intention;
        final String preData;
        Object postData;

        public MessagePacket (String[] intention, String controlSum, String data) {
            this.intention = intention;
            this.hash = controlSum;
            this.preData = data;
        }

        public void parseData (Class pattern) {
            postData = JsonIterator.deserialize(preData, pattern);
        }
    }

    @Contract("_ -> new")
    public static @NotNull ConnectionPath parseURI (@NotNull String uri) {
        int index = uri.indexOf('?');
        String path = uri.substring(0, index);
        String query = uri.substring(index + 1);

        String[] pathParts = path.split("/");

        Map<String, String> params = new HashMap<>();
        for (String param : query.split("&")) {
            String[] entry = param.split("=");

            params.put(entry[0], entry[1]);
        }

        return new ConnectionPath(pathParts, params);
    }

    @Contract("_ -> new")
    public static @NotNull MessagePacket parsePacket (@NotNull String packet) {
        String[] splitPacket = packet.split("%");

        return new MessagePacket(splitPacket[0].split("-"), splitPacket[1], splitPacket[2]);
    }

    public static Boolean verifierBCrypt (@NotNull String data, byte[] hash_data) {
        return BCrypt.verifyer().verify(Arrays.copyOfRange(data.toCharArray(), 0, Math.min(data.toCharArray().length, 72)), hash_data).verified;
    }

    public static String SHA512 (String string) {
        MessageDigest md;

        try {
            md = MessageDigest.getInstance("SHA-512");
        } catch (NoSuchAlgorithmException _) {
            return string;
        }

        byte[] bytes = md.digest(string.getBytes(StandardCharsets.UTF_8));
        StringBuilder sb = new StringBuilder();

        for (byte aByte : bytes) {
            sb.append(Integer.toString((aByte & 0xff) + 0x100, 16).substring(1));
        }

        return sb.toString();
    }
}

我正在测试:

public class Starter {
    public static void main (String[] args) {
        Helper.parseURI("/id=4&token=user.4.wsm3mOlDwbhaHD0iG33nQ9CwBEIf3B7al5Bylki1XpuKOHAxy2og1mmqdIs5yALmW");

        System.out.println("Прошло!");
    }
}

但程序崩溃了:

C:\Users\Fluorum\.jdks\sapmachine-21\bin\java.exe --enable-preview "-javaagent:C:\Users\Fluorum\AppData\Local\Programs\IntelliJ IDEA Community Edition\lib\idea_rt.jar=56493:C:\Users\Fluorum\AppData\Local\Programs\IntelliJ IDEA Community Edition\bin" -Dfile.encoding=UTF-8 -Dsun.stdout.encoding=UTF-8 -Dsun.stderr.encoding=UTF-8 -classpath E:\Projects\GigaChat\RTCD\out\production\RTCD;C:\Users\Fluorum\.m2\repository\com\jsoniter\jsoniter\0.9.23\jsoniter-0.9.23.jar;C:\Users\Fluorum\.m2\repository\org\java-websocket\Java-WebSocket\1.5.4\Java-WebSocket-1.5.4.jar;C:\Users\Fluorum\.m2\repository\org\slf4j\slf4j-api\2.0.6\slf4j-api-2.0.6.jar;C:\Users\Fluorum\.m2\repository\at\favre\lib\bcrypt\0.10.2\bcrypt-0.10.2.jar;C:\Users\Fluorum\.m2\repository\at\favre\lib\bytes\1.5.0\bytes-1.5.0.jar;C:\Users\Fluorum\.m2\repository\org\jetbrains\annotations\24.0.0\annotations-24.0.0.jar;C:\Users\Fluorum\.m2\repository\org\postgresql\postgresql\42.6.0\postgresql-42.6.0.jar;C:\Users\Fluorum\.m2\repository\org\checkerframework\checker-qual\3.31.0\checker-qual-3.31.0.jar Starter
Exception in thread "main" java.lang.ClassFormatError: Illegal field name "" in class Helper
    at java.base/java.lang.ClassLoader.defineClass1(Native Method)
    at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1027)
    at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:150)
    at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:862)
    at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:760)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:681)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:639)
    at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
    at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:526)
    at Starter.main(Starter.java:7)

Process finished with exit code 1

代码从哪里获取空字段?唯一的猜测是哈希映射键。但在这一行中它们不是空的?

UPD我试图进行标注:

public class Starter {
    @Contract("_ -> new")
    public static @NotNull Helper.ConnectionPath parseURI (@NotNull String uri) {
        int index = uri.indexOf('?');
        String path = uri.substring(0, index);
        String query = uri.substring(index + 1);

        String[] pathParts = path.split("/");

        System.out.println(Arrays.toString(pathParts));

        Map<String, String> params = new HashMap<>();
        for (String param : query.split("&")) {
            String[] entry = param.split("=");
            System.out.println("-----");
            System.out.println(Arrays.toString(entry));
            params.put(entry[0], entry[1]);
        }

        return new Helper.ConnectionPath(pathParts, params);
    }

    public static void main (String[] args) {
        Helper.ConnectionPath returner = parseURI("/id=4&token=user.4.wsm3mOlDwbhaHD0iG33nQ9CwBEIf3B7al5Bylki1XpuKOHAxy2og1mmqdIs5yALmW");

        System.out.println("Прошло!");
        System.out.println(returner);
    }
}

但现在错误完全不同了:

C:\Users\Fluorum\.jdks\sapmachine-21\bin\java.exe --enable-preview "-javaagent:C:\Users\Fluorum\AppData\Local\Programs\IntelliJ IDEA Community Edition\lib\idea_rt.jar=60335:C:\Users\Fluorum\AppData\Local\Programs\IntelliJ IDEA Community Edition\bin" -Dfile.encoding=UTF-8 -Dsun.stdout.encoding=UTF-8 -Dsun.stderr.encoding=UTF-8 -classpath E:\Projects\GigaChat\RTCD\out\production\RTCD;C:\Users\Fluorum\.m2\repository\com\jsoniter\jsoniter\0.9.23\jsoniter-0.9.23.jar;C:\Users\Fluorum\.m2\repository\org\java-websocket\Java-WebSocket\1.5.4\Java-WebSocket-1.5.4.jar;C:\Users\Fluorum\.m2\repository\org\slf4j\slf4j-api\2.0.6\slf4j-api-2.0.6.jar;C:\Users\Fluorum\.m2\repository\at\favre\lib\bcrypt\0.10.2\bcrypt-0.10.2.jar;C:\Users\Fluorum\.m2\repository\at\favre\lib\bytes\1.5.0\bytes-1.5.0.jar;C:\Users\Fluorum\.m2\repository\org\jetbrains\annotations\24.0.0\annotations-24.0.0.jar;C:\Users\Fluorum\.m2\repository\org\postgresql\postgresql\42.6.0\postgresql-42.6.0.jar;C:\Users\Fluorum\.m2\repository\org\checkerframework\checker-qual\3.31.0\checker-qual-3.31.0.jar Starter
Exception in thread "main" java.lang.StringIndexOutOfBoundsException: Range [0, -1) out of bounds for length 84
    at java.base/jdk.internal.util.Preconditions$1.apply(Preconditions.java:55)
    at java.base/jdk.internal.util.Preconditions$1.apply(Preconditions.java:52)
    at java.base/jdk.internal.util.Preconditions$4.apply(Preconditions.java:213)
    at java.base/jdk.internal.util.Preconditions$4.apply(Preconditions.java:210)
    at java.base/jdk.internal.util.Preconditions.outOfBounds(Preconditions.java:98)
    at java.base/jdk.internal.util.Preconditions.outOfBoundsCheckFromToIndex(Preconditions.java:112)
    at java.base/jdk.internal.util.Preconditions.checkFromToIndex(Preconditions.java:349)
    at java.base/java.lang.String.checkBoundsBeginEnd(String.java:4861)
    at java.base/java.lang.String.substring(String.java:2830)
    at Starter.parseURI(Starter.java:12)
    at Starter.main(Starter.java:35)

Process finished with exit code 1

UDP 2我又做了一些实验。我发现,如果我注释掉该方法SHA512(通过 的哈希函数SHA-512),错误就会消失并且一切正常。

java
  • 1 个回答
  • 45 Views
Martin Hope
mister_svinia
Asked: 2023-09-30 22:49:23 +0000 UTC

如何将 switch-case 与实际静态数据一起使用?

  • 6

有一个存储类是这样的:

public class CommandsIntentions {
    static Command ADMIN_CHANNELS_CREATE = new Command(new String[]{"ADMIN", "CHANNELS", "CREATE"}, DataCommands.ChannelCreate.class);
    static Command ADMIN_CHANNELS_USERS_JOIN = new Command(new String[]{"ADMIN", "CHANNELS", "USERS", "JOIN"}, DataCommands.ChannelUsersJoin.class);
    static Command ADMIN_CHANNELS_USERS_APPEND = null; // TODO
    static Command USER_CHANNELS_MESSAGES_POST_NEW = new Command(new String[]{"USER", "CHANNELS", "MESSAGES", "POST", "NEW"}, DataCommands.ChannelMessagesPostNew.class);
}

存储外壳是:

public class Command {
    String[] intents;
    Class pattern;

    public Command (String[] intents, Class pattern) {
        this.intents = intents;
        this.pattern = pattern;
    }
}

这是我的呼吁:

@Override
public void onMessage (WebSocket webSocket, String message) {
    Helper.MessagePacket packet = Helper.parsePacket(message);

    switch (packet.intention) {
        case CommandsIntentions.ADMIN_CHANNELS_CREATE.intents -> {
            break;
        }
        default -> {
            return;
        }
    }
}

但会抛出错误:

Constant expression required: 42

我还附上了包的解析:

public static class MessagePacket {
    final String hash;
    final String[] intention;
    final String preData;
    Object postData;

    public MessagePacket (String[] intention, String controlSum, String data) {
        this.intention = intention;
        this.hash = controlSum;
        this.preData = data;
    }

    public void parseData (Class pattern) {
        postData = JsonIterator.deserialize(preData, pattern);
    }
}

@Contract("_ -> new")
public static @NotNull MessagePacket parsePacket (@NotNull String packet) {
    String[] splitPacket = packet.split("%");

    return new MessagePacket(splitPacket[0].split("-"), splitPacket[1], splitPacket[2]);
}

主意

java
  • 1 个回答
  • 73 Views
Martin Hope
mister_svinia
Asked: 2023-09-21 20:01:20 +0000 UTC

自定义小部件不呈现

  • 5

我创建我的活动:

class ActivityMain : AppCompatActivity() {
    private var count = 0

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        binder()
    }


    private fun binder() {
        val button_addButton = findViewById<Button>(R.id.button_addButton)
        val button_addCustomWidget = findViewById<Button>(R.id.button_addCustomWidget)

        val layout = findViewById<LinearLayout>(R.id.testLayout)
        val scrollView = findViewById<ScrollView>(R.id.testScroll)

        val upButton = findViewById<Button>(R.id.upButton)
        val downButton = findViewById<Button>(R.id.downButton)

        upButton.setOnClickListener {
            scrollView.fullScroll(ScrollView.FOCUS_UP)
        }
        downButton.setOnClickListener {
            scrollView.fullScroll(ScrollView.FOCUS_DOWN)
        }

        button_addCustomWidget.setOnClickListener {
            val widget = ChannelMessageWidget(this, null, 0)

            widget.setLayoutParams(LinearLayout.LayoutParams(
                LinearLayout.LayoutParams.MATCH_PARENT,
                LinearLayout.LayoutParams.WRAP_CONTENT
            ))

            widget.setAvatar(
                BitmapFactory.decodeResource(getResources(),
                R.drawable.default_avatar))

            layout.addView(widget)
        }

        button_addButton.setOnClickListener {
            count ++

            val button = Button(this)

            button.text = "Кнопка $count!"

            button.setOnClickListener {
                Toast.makeText(this, "Привет, $count", Toast.LENGTH_SHORT).show()
            }

            layout.addView(button)
        }
    }
}

所有小部件均已初始化并出现一个窗口。这是它的 XML:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".ActivityMain">

    <Button
        android:id="@+id/button_addButton"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="AB"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent" />

    <ScrollView
        android:id="@+id/testScroll"
        android:layout_width="0dp"
        android:layout_height="0dp"
        app:layout_constraintBottom_toTopOf="@+id/button_addButton"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent">

        <LinearLayout
            android:id="@+id/testLayout"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical"
            tools:ignore="UselessLeaf" />
    </ScrollView>

    <Button
        android:id="@+id/upButton"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Up"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toStartOf="@+id/button_addCustomWidget"
        app:layout_constraintStart_toEndOf="@+id/downButton" />

    <Button
        android:id="@+id/downButton"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Down"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintStart_toStartOf="parent" />

    <Button
        android:id="@+id/button_addCustomWidget"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="ACW"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toStartOf="@+id/button_addButton"
        app:layout_constraintStart_toEndOf="@+id/upButton" />
</androidx.constraintlayout.widget.ConstraintLayout>

我还创建了一个自定义小部件(我正在尝试绘制圆形图片):

fun dp2pixel(dp: Float, context: Context): Float {
    return TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp, context.resources.displayMetrics)
}

class ChannelMessageWidget @JvmOverloads constructor(
    private val context: Context, attrs: AttributeSet? = null, defStyleAttr: Int
) : View(context, attrs, defStyleAttr) {
    private var text: String = "null"
    private lateinit var avatar: Bitmap
    private var displayed = false
    private val radius = 128f

//    @SuppressLint("DrawAllocation")
//    override fun onDraw(canvas: Canvas) {
//        super.onDraw(canvas)
//
//        val scaledBitmap = Bitmap.createScaledBitmap(avatar, dp2pixel(radius, context).toInt(), dp2pixel(radius, context).toInt(), true)
//
//        val outputBitmap = Bitmap.createBitmap(scaledBitmap.width, scaledBitmap.height, Bitmap.Config.ARGB_8888)
//
//        val paint = Paint().apply {
//            isAntiAlias = true
//            color = Color.BLACK
//            style = Paint.Style.FILL
//        }
//
//        val radius = scaledBitmap.width / 2f
//        canvas.drawCircle(radius, radius, radius, paint)
//
//        paint.xfermode = PorterDuffXfermode(PorterDuff.Mode.SRC_IN)
//        canvas.drawBitmap(outputBitmap, 0f, 0f, paint)
//
//        val rect = Rect(0f, 0f, scaledBitmap.width.toFloat(), scaledBitmap.height.toFloat())
//
//        displayed = true
//    }

//    @SuppressLint("DrawAllocation")
//    override fun onDraw(canvas: Canvas) {
//        super.onDraw(canvas)
//
//        val scaledBitmap = Bitmap.createScaledBitmap(avatar, dp2pixel(radius, context).toInt(), dp2pixel(radius, context).toInt(), true)
//
//        val outputBitmap = Bitmap.createBitmap(scaledBitmap.width, scaledBitmap.height, Bitmap.Config.ARGB_8888)
//        val outputCanvas = Canvas(outputBitmap) // Создаем Canvas для outputBitmap
//
//        val paint = Paint().apply {
//            isAntiAlias = true
//            color = Color.BLACK
//            style = Paint.Style.FILL
//        }
//
//        val radius = scaledBitmap.width / 2f
//        outputCanvas.drawCircle(radius, radius, radius, paint) // Рисуем круг на outputCanvas
//
//        paint.xfermode = PorterDuffXfermode(PorterDuff.Mode.SRC_IN)
//        outputCanvas.drawBitmap(scaledBitmap, 0f, 0f, paint) // Рисуем отмасштабированное изображение на outputCanvas
//
//        canvas.drawBitmap(outputBitmap, 0f, 0f, null) // Рисуем outputBitmap на холсте canvas
//    }

    @SuppressLint("DrawAllocation")
    override fun onDraw(canvas: Canvas) {
        super.onDraw(canvas)

        val paint = Paint().apply {
            isAntiAlias = true
            color = Color.BLACK
            style = Paint.Style.FILL
        }

        val rect = Rect(0, 0, 100, 50) // Прямоугольник с шириной экрана и высотой 50 пикселей

        canvas.drawRect(rect, paint) // Отрисовка прямоугольника на холсте
    }


    fun setAvatar(avatar: Bitmap) {
        this.avatar = avatar
    }

    fun setText(text: String) {
        this.text = text
    }
}

这些方法都不起作用。控制台不显示任何错误,我的自定义元素也没有添加到布局中。虽然按钮很容易添加、滚动等。

android
  • 1 个回答
  • 26 Views

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    我看不懂措辞

    • 1 个回答
  • Marko Smith

    请求的模块“del”不提供名为“default”的导出

    • 3 个回答
  • Marko Smith

    "!+tab" 在 HTML 的 vs 代码中不起作用

    • 5 个回答
  • Marko Smith

    我正在尝试解决“猜词”的问题。Python

    • 2 个回答
  • Marko Smith

    可以使用哪些命令将当前指针移动到指定的提交而不更改工作目录中的文件?

    • 1 个回答
  • Marko Smith

    Python解析野莓

    • 1 个回答
  • Marko Smith

    问题:“警告:检查最新版本的 pip 时出错。”

    • 2 个回答
  • Marko Smith

    帮助编写一个用值填充变量的循环。解决这个问题

    • 2 个回答
  • Marko Smith

    尽管依赖数组为空,但在渲染上调用了 2 次 useEffect

    • 2 个回答
  • Marko Smith

    数据不通过 Telegram.WebApp.sendData 发送

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5