android设置添加设备QR码信息

摘要:客户衍生需求,通过扫QR码快速获取设备基础信息,并且基于POS SDK进行打印。

1. 定位至device info的xml添加相关perference

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml	(版本 347)
@@ -92,10 +92,17 @@android:selectable="false"android:title="@string/my_device_info_device_details_category_title">+        <com.android.settings.widget.QRDeviceInfoPreference
+            android:key="qr_info"
+            android:order="18"
+            android:persistent="false"
+            android:selectable="false"
+            settings:controller="com.android.settings.deviceinfo.QRDeviceInfoPreferenceController" />
+<!-- SIM status --><Preferenceandroid:key="sim_status"
-            android:order="18"
+            android:order="19"android:title="@string/sim_status_title"settings:keywords="@string/keywords_sim_status"android:summary="@string/summary_placeholder"
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/aboutphone/MyDeviceInfoFragment.java
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/aboutphone/MyDeviceInfoFragment.java	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/aboutphone/MyDeviceInfoFragment.java	(版本 347)
@@ -41,6 +41,7 @@
+import com.android.settings.deviceinfo.QRDeviceInfoPreferenceController;
@@ -150,6 +151,7 @@
+        controllers.add(new QRDeviceInfoPreferenceController(context));return controllers;}

2. preference布局

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/layout/qr_device_info_preference.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/layout/qr_device_info_preference.xml	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/layout/qr_device_info_preference.xml	(版本 347)
@@ -0,0 +1,23 @@
+<?xml version="1.0" encoding="utf-8"?>
+<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:orientation="vertical" >
+
+    <ImageView
+        android:id="@+id/iv_qr_image"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:layout_gravity="center"/>
+
+    <Button
+        android:id="@+id/print_button"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:layout_gravity="bottom|end"
+        android:paddingVertical="14dp"
+        android:drawableStart="@drawable/ic_print"
+        android:drawablePadding="9dp"
+        android:text="@string/print"
+        style="@style/ActionPrimaryButton"/>
+</FrameLayout>
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/drawable/ic_print.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/drawable/ic_print.xml	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/drawable/ic_print.xml	(版本 347)
@@ -0,0 +1,26 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+  ~ Copyright 2017 The Android Open Source Project
+  ~
+  ~ Licensed under the Apache License, Version 2.0 (the "License");
+  ~ you may not use this file except in compliance with the License.
+  ~ You may obtain a copy of the License at
+  ~
+  ~      http://www.apache.org/licenses/LICENSE-2.0
+  ~
+  ~ Unless required by applicable law or agreed to in writing, software
+  ~ distributed under the License is distributed on an "AS IS" BASIS,
+  ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+  ~ See the License for the specific language governing permissions and
+  ~ limitations under the License.
+  -->
+
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24.0"
+    android:viewportHeight="24.0">
+    <path
+        android:pathData="M19,8L5,8c-1.66,0 -3,1.34 -3,3v6h4v4h12v-4h4v-6c0,-1.66 -1.34,-3 -3,-3zM16,19L8,19v-5h8v5zM19,12c-0.55,0 -1,-0.45 -1,-1s0.45,-1 1,-1 1,0.45 1,1 -0.45,1 -1,1zM18,3L6,3v4h12L18,3z"
+        android:fillColor="#FFFFFF"/>
+</vector>

3. 实现

QR码的生成使用的是google公开的com.google.zxing

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreference.java
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreference.java	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreference.java	(版本 347)
@@ -0,0 +1,401 @@
+/*
+ * Copyright (C) 2017 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package com.android.settings.widget;
+
+import android.app.AlertDialog;
+import android.content.Context;
+import android.content.DialogInterface;
+import android.graphics.Bitmap;
+import android.graphics.BitmapFactory;
+import android.graphics.Canvas;
+import android.os.Build;
+import android.os.Environment;
+import android.os.Handler;
+import android.os.Message;
+import android.os.SystemProperties;
+import android.telephony.TelephonyManager;
+import android.text.TextUtils;
+import android.util.AttributeSet;
+import android.util.Log;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.Button;
+import android.widget.ImageView;
+
+import androidx.preference.Preference;
+import androidx.preference.PreferenceViewHolder;
+
+import com.android.settings.R;
+
+import com.google.zxing.BarcodeFormat;   
+import com.google.zxing.EncodeHintType;   
+import com.google.zxing.MultiFormatWriter;   
+import com.google.zxing.WriterException;   
+import com.google.zxing.common.BitMatrix;
+
+import com.pos.sdk.accessory.PosAccessoryManager;
+import com.pos.sdk.printer.PosPrinter;
+
+import java.lang.StringBuilder;
+import java.util.Hashtable;
+
+public class QRDeviceInfoPreference extends Preference {
+
+    private static final String TAG = "QRDeviceInfoPreference";
+
+    private static final int BLACK = 0xff000000;
+
+    private static final int PRINTER_ERROR_NO_PAPER = 1;
+    private static final int PRINTER_ERROR_OVER_HEAT = 2;
+    private static final int PRINTER_STATE_CHANGED = 3;
+    private static final int PRINTER_ERROR_NO_BATTERY = 4;
+
+    private Context mContext;
+    private Bitmap qrCodeBitmap;
+    private Button mPrintBtn;
+    private ImageView qrImgImageView;
+    private PosPrinter mPrinter;
+    private TelephonyManager mTelephonyManager;
+    private String mSpVersion, mDsn, mPsn;
+
+    private PosPrinter.EventListener mListener = new PosPrinter.EventListener() {
+        @Override
+        public void onInfo(PosPrinter printer, int what, int extra) {
+            Log.i(TAG, "onInfo: what= " + what + ", extra= " + extra);
+            if (what == PosPrinter.PRINTER_INFO_STATE_CHANGED) {
+                mHandler.sendMessage(mHandler.obtainMessage(PRINTER_STATE_CHANGED, extra, 0));
+            }
+        }
+
+        @Override
+        public void onError(PosPrinter printer, int what, int extra) {
+            switch(what) {
+                case PosPrinter.PRINTER_ERROR_NO_PAPER:
+                    mHandler.sendEmptyMessage(PRINTER_ERROR_NO_PAPER);
+                    break;
+                case PosPrinter.PRINTER_ERROR_OVER_HEAT:
+                    Log.e(TAG, "PRINTER_ERROR_OVER_HEAT");
+                    mHandler.sendEmptyMessage(PRINTER_ERROR_OVER_HEAT);
+                    break;
+                case PosPrinter.PRINTER_ERROR_STATE:
+                    Log.e(TAG, "PRINTER_ERROR_STATE");
+                    if (extra == PosPrinter.PRINTER_STATE_NO_BATTERY) {
+                        mHandler.sendEmptyMessage(PRINTER_ERROR_NO_BATTERY);
+                    }
+                    break;
+                default:
+                    Log.e(TAG, "PRINTER_ERROR: " + what);
+                    break;
+            }
+        }
+
+        @Override
+        public void onCursorChanged(PosPrinter printer, int x, int y, int lastX, int lastY) {
+            Log.d(TAG, "onCursorChanged: x= " + x + ", y= " + ", lastX= " + ", lastY=" + lastY);
+        }
+    };
+
+    private Handler mHandler = new Handler() {
+        public void handleMessage(Message msg) {
+            super.handleMessage(msg);
+            switch (msg.what) {
+                case PRINTER_ERROR_NO_PAPER: {
+                    AlertDialog.Builder dlg = new AlertDialog.Builder(mContext);
+                    dlg.setTitle(R.string.reload_paper);
+                    dlg.setCancelable(false);
+                    dlg.setPositiveButton(R.string.been_installed_paper, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(false);
+                            mHandler.post(mPrinterRunnable);
+                        }
+                    });
+                    dlg.setNegativeButton(R.string.dlg_cancel, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(true);
+                        }
+                    });
+                    dlg.show();
+                    break;
+                }
+                case PRINTER_ERROR_OVER_HEAT: {
+                    AlertDialog.Builder dlg = new AlertDialog.Builder(mContext);
+                    dlg.setTitle(R.string.printer_overheat);
+                    dlg.setMessage(R.string.printer_overheat_summary);
+                    dlg.setCancelable(false);
+                    dlg.setPositiveButton(R.string.ok, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(true);
+                        }
+                    });
+                    dlg.show();
+                    break;
+                }
+                case PRINTER_ERROR_NO_BATTERY: {
+                    AlertDialog.Builder dlg = new AlertDialog.Builder(mContext);
+                    dlg.setTitle(R.string.printer_no_battery);
+                    dlg.setMessage(R.string.printer_no_battery_summary);
+                    dlg.setCancelable(false);
+                    dlg.setPositiveButton(R.string.ok, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(true);
+                        }
+                    });
+                    dlg.show();
+                    break;
+                }
+                case PRINTER_STATE_CHANGED:
+                    if (msg.arg1 == PosPrinter.PRINTER_STATE_PRINTING) {
+                        mPrintBtn.setEnabled(false);
+                    } else {
+                        mPrintBtn.setEnabled(true);
+                    }
+                    break;
+                default:
+                    break;
+            }
+        }
+    };
+
+    public QRDeviceInfoPreference(Context context, AttributeSet attributeSet) {
+        this(context, attributeSet, 0);
+    }
+
+    public QRDeviceInfoPreference(Context context, AttributeSet attributeSet, int paramInt) {
+        super(context, attributeSet, paramInt);
+        setLayoutResource(R.layout.qr_device_info_preference);
+        mContext = context;
+        mTelephonyManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);
+        mDsn = mTelephonyManager.getDsn(mContext);
+        mPsn = mTelephonyManager.getPsn(mContext);
+        mSpVersion = PosAccessoryManager.getDefault().getSpVersion();
+
+        int ret = -1;
+        int numOfPrinter = PosPrinter.getNumberOfPrinters();
+        if (numOfPrinter > 0) {
+            mPrinter = PosPrinter.open();
+            if (mPrinter != null) {
+                mPrinter.setOnEventListener(mListener);
+                ret = 0;
+            }
+        }
+        if (ret == -1) {
+            Log.d(TAG, "no printer");
+        }
+    }
+
+    @Override
+    public void onBindViewHolder(PreferenceViewHolder view) {
+        super.onBindViewHolder(view);
+
+        mPrintBtn = (Button) view.findViewById(R.id.print_button);
+        mPrintBtn.setOnClickListener(new View.OnClickListener() {
+            @Override
+            public void onClick(View v) {
+                mHandler.post(mPrinterRunnable);
+            }
+        });
+
+        qrImgImageView = (ImageView) view.findViewById(R.id.iv_qr_image);
+        qrImgImageView.setEnabled(false);
+        qrImgImageView.setOnClickListener(new View.OnClickListener() {
+            @Override
+            public void onClick(View v) {
+                showDialog();
+            }
+        });
+        qrImgImageView.setOnLongClickListener(new View.OnLongClickListener() {
+            @Override
+            public boolean onLongClick(View v) {
+                if (qrCodeBitmap != null) {
+                    print(qrCodeBitmap);
+                }
+                return true;
+            }
+        });
+
+        try {
+            String contentString = getDeviceInfo();
+            if (!TextUtils.isEmpty(contentString)) {
+                qrCodeBitmap = createQRCode(contentString, 350);
+                qrImgImageView.setImageBitmap(qrCodeBitmap);
+            }
+        } catch (WriterException e) {
+            // TODO Auto-generated catch block
+            e.printStackTrace();
+        }
+    }
+
+    private Bitmap createQRCode(String str,int widthAndHeight) throws WriterException {
+        Hashtable<EncodeHintType, String> hints = new Hashtable<EncodeHintType, String>();
+        hints.put(EncodeHintType.CHARACTER_SET, "utf-8");
+        BitMatrix matrix = new MultiFormatWriter().encode(str,
+                BarcodeFormat.QR_CODE, widthAndHeight, widthAndHeight);
+        int width = matrix.getWidth();
+        int height = matrix.getHeight();
+        int[] pixels = new int[width * height];
+
+        for (int y = 0; y < height; y++) {
+            for (int x = 0; x < width; x++) {
+                if (matrix.get(x, y)) {
+                    pixels[y * width + x] = BLACK;
+                }
+            }
+        }
+        Bitmap bitmap = Bitmap.createBitmap(width, height,
+                Bitmap.Config.ARGB_8888);
+        bitmap.setPixels(pixels, 0, width, 0, 0, width, height);
+
+        // add logo
+        Bitmap brandBitmap = BitmapFactory.decodeResource(mContext.getResources(), R.mipmap.company_logo);
+        if (brandBitmap != null) {
+            bitmap = addLogoToQRCode(bitmap, brandBitmap);
+        }
+        return bitmap;
+    }
+
+    private Bitmap addLogoToQRCode(Bitmap qrCodeBitmap, Bitmap logoBitmap) {
+        if (qrCodeBitmap == null) {
+            Log.d(TAG, "addLogoToQRCode::qrCodeBitmap is null");
+            return null;
+        }
+
+        if (logoBitmap == null) {
+            Log.d(TAG, "addLogoToQRCode::logoBitmap is null");
+            return qrCodeBitmap;
+        }
+
+        int qrWidth = qrCodeBitmap.getWidth();
+        int qrHeight = qrCodeBitmap.getHeight();
+        int logoWidth = logoBitmap.getWidth();
+        int logoHeight = logoBitmap.getHeight();
+
+        if (qrWidth == 0 || qrHeight == 0) {
+            Log.d(TAG, "addLogoToQRCode::qrCodeBitmap size is wrong");
+            return null;
+        }
+
+        if (logoWidth == 0 || logoHeight == 0) {
+            Log.d(TAG, "addLogoToQRCode::logoBitmap size is wrong");
+            return qrCodeBitmap;
+        }
+
+        float scaleFactor = qrWidth * 1.0f / 5 / logoWidth;
+        Bitmap resultBitmap = Bitmap.createBitmap(qrWidth, qrHeight, qrCodeBitmap.getConfig());
+        Canvas canvas = new Canvas(resultBitmap);
+        canvas.drawBitmap(qrCodeBitmap, 0, 0, null);
+
+        canvas.scale(scaleFactor, scaleFactor, qrWidth / 2, qrHeight / 2);
+        canvas.drawBitmap(logoBitmap, (qrWidth - logoWidth) / 2, (qrHeight - logoHeight) / 2, null);
+        canvas.save(Canvas.ALL_SAVE_FLAG);
+        canvas.restore();
+
+        return resultBitmap;
+    }
+
+    private void print(Bitmap bitmap) {
+        int num = PosPrinter.getNumberOfPrinters();
+        if (num > 0) {
+            mPrinter = PosPrinter.open();
+            mPrinter.printBitmap(bitmap);
+        }
+    }
+
+    private void showDialog() {
+        AlertDialog.Builder builder = new AlertDialog.Builder(mContext);
+        final View dialogView = LayoutInflater.from(mContext).inflate(R.layout.qr_device_info_preference, null);
+        ImageView mImageView = (ImageView) dialogView.findViewById(R.id.iv_qr_image);
+        mImageView.setImageBitmap(qrCodeBitmap);
+        builder.setView(dialogView);
+        builder.setPositiveButton(R.string.print_settings, new DialogInterface.OnClickListener() { 
+            @Override 
+            public void onClick(DialogInterface dialog, int which) {
+                if (qrCodeBitmap != null) {
+                    print(qrCodeBitmap);
+                }
+            } 
+        });
+        builder.show();
+    }
+
+    private String getDeviceInfo() {
+        String model = "Model:" + Build.MODEL;
+        String dsn = "DSN:" + mDsn;
+        String psn = "PSN:" + mPsn;
+        String ap_version = "AP:" + SystemProperties.get("ro.build.sw.version");
+        String sp_version = "SP:" + mSpVersion;
+        String modem_version = "Modem:" + Build.getRadioVersion();
+        StringBuilder sb = new StringBuilder();
+        sb.append(model + "\n");
+        sb.append(dsn + "\n");
+        sb.append(psn + "\n");
+        sb.append(ap_version + "\n");
+        sb.append(sp_version + "\n");
+        sb.append(modem_version + "\n");
+        return sb.toString();
+    }
+
+    private Runnable mPrinterRunnable = new Runnable(){
+        @Override
+        public void run() {
+            mPrinter.cleanCache();
+            PosPrinter.Parameters param = mPrinter.getParameters();
+
+            param.setPrintAlign(param.ALIGH_CENTER);
+            mPrinter.setParameters(param);
+            Bitmap brandBitmap = BitmapFactory.decodeResource(mContext.getResources(), R.mipmap.company_logo);
+            if (brandBitmap != null) {
+                mPrinter.addBitmapToCache(brandBitmap);
+            }
+
+            param.setFontSize(30);
+            param.setPrintAlign(param.ALIGH_CENTER);
+            param.setFontName(Environment.getRootDirectory().toString() + "/fonts/DroidSansMono.ttf");
+            param.setFontEffet(param.EFFECT_BOLD);
+            mPrinter.setParameters(param);
+            mPrinter.addTextToCurCache("---" + mContext.getResources().getString(R.string.device_info) + "---");
+            
+            param.setFontSize(24);
+            param.setPrintAlign(param.ALIGH_LEFT);
+            param.setFontName(Environment.getRootDirectory().toString() + "/fonts/Roboto-Italic.ttf");
+            param.setFontEffet(param.EFFECT_NONE);
+            mPrinter.setParameters(param);
+            mPrinter.addTextToCurCache("Model: " + Build.MODEL);
+            mPrinter.addTextToCurCache("AP: " + SystemProperties.get("ro.build.sw.version"));
+            mPrinter.addTextToCurCache("SP: " + mSpVersion);
+            mPrinter.addTextToCurCache("DSN: " + mDsn);
+            mPrinter.addTextToCurCache("PSN: " + mPsn);
+            mPrinter.addTextToCurCache("Modem: " + Build.getRadioVersion());
+
+            param.setFontSize(18);
+            param.setPrintAlign(param.ALIGH_CENTER);
+            param.setFontName(Environment.getRootDirectory().toString() + "/fonts/DroidSansMono.ttf");
+            param.setFontEffet(param.EFFECT_LEAN);
+            mPrinter.setParameters(param);
+            mPrinter.addTextToCurCache("-----" + mContext.getResources().getString(R.string.website) + "-----");
+
+            if (qrCodeBitmap != null) {
+                mPrinter.addBitmapToCache(qrCodeBitmap);
+            }
+            mPrinter.addTextToCurCache("\n\n\n");
+            mPrinter.print();
+        }
+    };
+}

4. 控制preference是否显示

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/config.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/config.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/config.xml	(版本 347)
@@ -649,4 +649,6 @@+
+    <bool name="config_show_qr_info">true</bool></resources>
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreferenceController.java
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreferenceController.java	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreferenceController.java	(版本 347)
@@ -0,0 +1,35 @@
+package com.android.settings.deviceinfo;
+
+import android.content.Context;
+import androidx.preference.PreferenceScreen;
+
+import com.android.settings.R;
+import com.android.settings.core.BasePreferenceController;
+import com.android.settings.widget.QRDeviceInfoPreference;
+
+public class QRDeviceInfoPreferenceController extends BasePreferenceController {
+    private static final String PREF_KEY = "qr_info";
+    private QRDeviceInfoPreference mPreference;
+
+    public QRDeviceInfoPreferenceController(Context context) {
+        super(context, PREF_KEY);
+    }
+
+    @Override
+    public void displayPreference(PreferenceScreen screen) {
+        super.displayPreference(screen);
+        mPreference = (QRDeviceInfoPreference) screen.findPreference(PREF_KEY);
+    }
+
+    @Override
+    public int getAvailabilityStatus() {
+        return mContext.getResources()
+                .getBoolean(R.bool.config_show_qr_info)
+                ? AVAILABLE : DISABLED_FOR_USER;
+    }
+
+    @Override
+    public String getPreferenceKey() {
+        return PREF_KEY;
+    }
+}

5. 相关字串

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/strings.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/strings.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/strings.xml	(版本 347)
@@ -364,4 +364,15 @@
+
+    <string name="print">print</string>
+    <string name="reload_paper">reload paper</string>
+    <string name="dlg_cancel">cancel</string>
+    <string name="been_installed_paper">paper ready</string>
+    <string name="printer_overheat">printer overheat</string>
+    <string name="printer_overheat_summary">printer overheat, please try again later</string>
+    <string name="printer_no_battery">printer no battery</string>
+    <string name="printer_no_battery_summary">no battery, please install the battery and try again</string>
+    <string name="device_info">DEVICE INFO</string>
+    <string name="website">www.xxx.cn</string></resources>
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values-zh-rCN/strings.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values-zh-rCN/strings.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values-zh-rCN/strings.xml	(版本 347)
@@ -386,4 +386,15 @@
+    <string name="print">打印</string>
+    <string name="reload_paper">请重新装纸</string>
+    <string name="dlg_cancel">"取消"</string>
+    <string name="been_installed_paper">已经装好纸</string>
+    <string name="printer_overheat">温度过热</string>
+    <string name="printer_overheat_summary">打印机温度过热,请稍候再试</string>
+    <string name="printer_no_battery">没有电池</string>
+    <string name="printer_no_battery_summary">未安装电池,请安装电池再试</string>
+    <string name="no_printer">没有找到打印机</string>
+    <string name="no_printer_summary">没有找到打印机,请检查打印硬件是否正常</string>
+    <string name="device_info">设备信息</string></resources>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/69715.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Ubuntu 多版本 gcc 配置常用命令备忘

用的频率不高&#xff0c;总忘记具体参数 1&#xff0c;安装多版本 gcc 以 gcc-11 和12 为例&#xff1a; sudo apt-get install gcc-11 gcc-12 sudo apt-get install gcc-11 gcc-12 2&#xff0c;配置多版本 gcc gcc 与 g 一起配置进数据库中&#xff1a; sudo update-a…

【kubernetes组件合集】深入解析Kubernetes组件之三:client-go

深入解析Kubernetes组件之三&#xff1a;client-go 目录 深入解析Kubernetes组件之三&#xff1a;client-go 引言 1. client-go简介 2. client-go的功能 2.1 资源操作 2.2 资源监听 2.3 认证和授权 2.4 错误处理和重试 2.5 扩展性和定制化 3. 使用client-go与Kubern…

多项式插值(数值计算方法)Matlab实现

多项式插值&#xff08;数值计算方法&#xff09;Matlab实现 一. 原理介绍二. 程序设计1. 构建矩阵2. 求解矩阵方程3. 作出多项式函数4. 绘制插值曲线5. 完整代码 三. 图例 一. 原理介绍 关于插值的定义及基本原理可以参照如下索引 插值原理&#xff08;数值计算方法&#xff…

vite + axios 代理不起作用 404 无效

vite axios 代理不起作用 先看官方示例 export default defineConfig({server: {proxy: {// 字符串简写写法/foo: http://localhost:4567,// 选项写法/api: {target: http://jsonplaceholder.typicode.com,changeOrigin: true,rewrite: (path) > path.replace(/^\/api/, )…

国产编辑器EverEdit - 编辑辅助功能介绍

1 编辑辅助功能 1.1 各编辑辅助选项说明 1.1.1 行号 打开该选项时&#xff0c;在编辑器主窗口左侧显示行号&#xff0c;如下图所示&#xff1a; 1.1.2 文档地图 打开该选项时&#xff0c;在编辑器主窗口右侧靠近垂直滚动条的地方显示代码的缩略图&#xff0c;如下图所示&…

深入理解Java对接DeepSeek

其实&#xff0c;整个对接过程很简单&#xff0c;就四步&#xff0c;获取key&#xff0c;找到接口文档&#xff0c;接口测试&#xff0c;代码对接。 1.获取 KEY https://platform.deepseek.com/transactions 直接付款就是了&#xff08;现在官网暂停充值2025年2月7日&#xf…

调用DeepSeek官方的API接口

效果 前端样式体验链接&#xff1a;https://livequeen.top/deepseekshow 准备工作 1、注册deepseek官网账号 地址&#xff1a;DeepSeek 点击进入右上角【API开放平台】&#xff0c;并进行账号注册。 2、注册完成后&#xff0c;依次点击【API keys】-【生成API key】&#x…

香港中文大学 Adobe 推出 MotionCanvas:开启用户掌控的电影级图像视频创意之旅。

简介&#xff1a; 亮点直击 将电影镜头设计引入图像到视频的合成过程中。 推出了MotionCanvas&#xff0c;这是一种简化的视频合成系统&#xff0c;用于电影镜头设计&#xff0c;提供整体运动控制&#xff0c;以场景感知的方式联合操控相机和对象的运动。 设计了专门的运动条…

01.Docker 概述

Docker 概述 1. Docker 的主要目标2. 使用Docker 容器化封装应用程序的意义3. 容器和虚拟机技术比较4. 容器和虚拟机表现比较5. Docker 的组成6. Namespace7. Control groups8. 容器管理工具9. docker 的优缺点10. 容器的相关技术 docker 官网: http://www.docker.com 帮助文档…

【DeepSeek】deepseek可视化部署

目录 1 -> 前文 2 -> 部署可视化界面 1 -> 前文 【DeepSeek】DeepSeek概述 | 本地部署deepseek 通过前文可以将deepseek部署到本地使用&#xff0c;可是每次都需要winR输入cmd调出命令行进入到命令模式&#xff0c;输入命令ollama run deepseek-r1:latest。体验很…

开启对话式智能分析新纪元——Wyn商业智能 BI 携手Deepseek 驱动数据分析变革

2月18号&#xff0c;Wyn 商业智能 V8.0Update1 版本将重磅推出对话式智能分析&#xff0c;集成Deepseek R1大模型&#xff0c;通过AI技术的深度融合&#xff0c;致力于打造"会思考的BI系统"&#xff0c;让数据价值触手可及&#xff0c;助力企业实现从数据洞察到决策执…

Response 和 Request 介绍

怀旧网个人博客网站地址&#xff1a;怀旧网&#xff0c;博客详情&#xff1a;Response 和 Request 介绍 1、HttpServletResponse 1、简单分类 2、文件下载 通过Response下载文件数据 放一个文件到resources目录 编写下载文件Servlet文件 public class FileDownServlet exten…

分层耦合 - IOC详解

推荐使用下面三种, 第一种多用于其他类 声明bean的时候&#xff0c;可以通过value属性指定bean的名字&#xff0c;如果没有指定&#xff0c;默认为类名首字母小写。 使用以上四个注解都可以声明bean&#xff0c;但是在springboot集成web开发中&#xff0c;声明控制器bean只能用…

STM32 Flash详解教程文章

目录 Flash基本概念理解 Flash编程接口FPEC Flash擦除/写入流程图 Flash选项字节基本概念理解 Flash电子签名 函数读取地址下存放的数据 Flash的数据处理限制部分 编写不易&#xff0c;请勿搬运&#xff0c;感谢理解&#xff01;&#xff01;&#xff01; Flash基本概念…

WPF 设置宽度为 父容器 宽度的一半

方法1&#xff1a;使用 绑定和转换器 实现 创建类文件 HalfWidthConverter public class HalfWidthConverter : IValueConverter{public object Convert(object value, Type targetType, object parameter, CultureInfo culture){if (value is double width){return width / 4…

【Ubuntu VScode Remote SSH 问题解决】Resolver error: Error: XHR failed

1. 问题描述 VScode使用remote ssh 远程服务器&#xff0c;报错类似&#xff1a; [12:06:01.219] Downloading VS Code server locally... [12:06:01.310] Resolver error: Error: XHR failedat k.onerror (vscode-file://vscode-app/private/var/folders/g1/cvs2rnpx60qc3b4…

32单片机学习记录1之GPIO

32单片机学习记录1之GPIO 前置 GPIO口在单片机中扮演着什么角色&#xff1f; 在单片机中&#xff0c;GPIO口&#xff08;General Purpose Input/Output&#xff09; 是一种通用输入/输出接口&#xff0c;扮演着连接单片机与外部设备的桥梁角色。具体来说&#xff0c;它在单片…

第三十二周:Informer学习笔记

目录 摘要Abstract1 Informer1.1 预备知识1.2 模型框架1.3 实验分析 总结 摘要 本周学习的主要内容是Informer模型&#xff0c;Informer是一种专为长序列时间序列预测&#xff08;LSTF&#xff09; 设计的Transformer模型。相较于传统的Transformer&#xff0c;Informer采用Pr…

绩效归因概述

绩效归因概述 1. 分类2. 基于净值的归因方法2.1 发展背景2.2 择时选股模型 T-M模型2.3 择时选股模型 H-M模型2.4 择时选股模型 C-L模型2.5 风格配置模型-Sharpe2.6 多因子模型 Fama-French32.7 多因子模型 Carhart42.8 多因子模型 Fama-French5 3. 基于持仓的归因方法3.1 发展背…

MambaMorph brain MR-CT

loss代码实现了几种用于医学图像配准(Registration)和分割(Segmentation)任务的损失函数,主要包括以下几种: NCC (Normalized Cross-Correlation): 功能: 计算局部归一化互相关损失,用于衡量两个图像之间的相似性。 应用场景: 通常用于图像配准任务,通过最大化图像之间…