wpf 模擬抖音很火的羅盤時鐘,附源碼,下載就能跑

来源:https://www.cnblogs.com/xuling-297769461/archive/2019/05/10/10845590.html
-Advertisement-
Play Games

wpf 模擬抖音很火的羅盤時鐘,附源碼 前端時間突然發現,抖音火了個壁紙,就是黑底蕾絲~~~ 錯錯錯,黑底白字的羅盤時鐘! 作為程式員的我,也覺得很新穎,所以想空了研究下,這不,空下來了就用wpf,寫個屬於.net自己的羅盤時鐘,目前只實現了時分秒,農曆日期等邏輯都是一樣的,所以就略了,有興趣的朋友 ...


wpf 模擬抖音很火的羅盤時鐘,附源碼

  前端時間突然發現,抖音火了個壁紙,就是黑底蕾絲~~~  錯錯錯,黑底白字的羅盤時鐘

  作為程式員的我,也覺得很新穎,所以想空了研究下,這不,空下來了就用wpf,寫個屬於.net自己的羅盤時鐘,目前只實現了時分秒,農曆日期等邏輯都是一樣的,所以就略了,有興趣的朋友,可以繼續深入!

  最開始想直接弄成成exe,方便拷貝,到處運行使用的,但是考慮到,萬一有網友朋友們需要,所以我還是把封成一個dll,需要的地方添加引用即可!

  為了弄這個,還惡補了下,高中還是初中的知識,sin30,cos60,呵呵,正弦餘弦,所以不明白的朋友們需要先瞭解清楚這個,因為羅盤是旋轉,需要用到計算這個值!

  不廢話了,先上圖看下效果!

 

 

 

  

   ok,整體效果就是這樣了,中間是鄙人的名稱縮寫,抖音上是很潦草的,我就隨便啦,占個位置,不然顯得很空洞!

  下麵說說代碼

 

 

  主要是,RomeClockControlLibrary,這個是對控制項的封裝,上面那個啟動程式只是一個容器,或者說是調用者,當然,如果要達到我這個效果,實現圓形的視窗透明的朋友們,可以看下借鑒!

 

<UserControl x:Class="RomeClockControlLibrary.RomeClockControl"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:local="clr-namespace:RomeClockControlLibrary"
             mc:Ignorable="d" 
             d:DesignHeight="450" 
             d:DesignWidth="800"
             >
    <Border x:Name="bor"
            Background="#000000"
            >
        <Grid x:Name="grid" >
        </Grid>
    </Border>
</UserControl>

  

  上面是前端代碼,有點基礎的都應該看得懂,沒什麼可說的

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;

namespace RomeClockControlLibrary
{
    /// <summary>
    /// 羅馬時鐘
    /// </summary>
    public partial class RomeClockControl : UserControl, IDisposable
    {
        public RomeClockControl()
        {
            InitializeComponent();
        }

        /// <summary>
        /// x軸的中心位置
        /// </summary>
        private double CenterPixToX => this.ActualWidth / 2;

        /// <summary>
        /// y軸的中心位置
        /// </summary>
        private double CenterPixToY => this.ActualHeight / 2;

        /// <summary>
        /// 秒
        /// </summary>
        private Canvas CanvasHour = null;

        /// <summary>
        /// 分
        /// </summary>
        private Canvas CanvasMinute = null;

        /// <summary>
        /// 時
        /// </summary>
        private Canvas CanvasSecond = null;

        /// <summary>
        /// UI更新線程
        /// </summary>
        private Thread thread = null;

        /// <summary>
        /// 緩存時的顯示控制項
        /// </summary>
        private TextBlock BlockHour = null;

        /// <summary>
        /// 緩存分的顯示控制項
        /// </summary>
        private TextBlock BlockMinute = null;

        /// <summary>
        /// 緩存秒的顯示控制項
        /// </summary>
        private TextBlock BlockSecond = null;

        /// <summary>
        /// 添加控制項
        /// </summary>
        private void Add(AddType type)
        {
            var offset = 0;//偏移量
            var count = 0;//總量
            var str = string.Empty;
            var time = 0;
            double AngleTime = 0;
            Canvas canvas = new Canvas();
            canvas.Tag = type;

            switch (type)
            {
                case AddType.Hour:
                    offset = 95;
                    count = 24;
                    str = "時";
                    CanvasHour = canvas;
                    time = DateTime.Now.Hour;
                    break;
                case AddType.Minute:
                    offset = 60;
                    count = 60;
                    str = "分";
                    CanvasMinute = canvas;
                    time = DateTime.Now.Minute;
                    break;
                case AddType.Second:
                    offset = 30;
                    count = 60;
                    str = "秒";
                    CanvasSecond = canvas;
                    time = DateTime.Now.Second;
                    break;
                default:
                    return;
            }

            var angle = 360 / count;//角度
            var x = CenterPixToX - offset;//起始位置
            var y = CenterPixToY - offset;

            for (int i = 0; i < count; i++)
            {
                TextBlock t = new TextBlock();
                if (i <= 9)
                {
                    t.Text = $"0{i}{str}";
                }
                else
                {
                    t.Text = $"{i}{str}";
                }
                t.Tag = i;
                t.Foreground = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#7d7d7d"));
                canvas.Children.Add(t);

                var sinv = Math.Sin((90 - angle * i) * (Math.PI / 180));
                var cosv = Math.Cos((90 - angle * i) * (Math.PI / 180));
                var a = CenterPixToY - y * sinv;
                var b = CenterPixToX + y * cosv;
                Canvas.SetLeft(t, b);
                Canvas.SetTop(t, a);

                //設置角度
                RotateTransform r = new RotateTransform();
                r.Angle = angle * i - 90;
                t.RenderTransform = r;

                if (i == time)
                {
                    AngleTime = 360 - r.Angle;
                    //更新樣式
                    t.Foreground = new SolidColorBrush(Colors.White);
                    switch (type)
                    {
                        case AddType.Hour:
                            BlockHour = t;
                            break;
                        case AddType.Minute:
                            BlockMinute = t;
                            break;
                        case AddType.Second:
                            BlockSecond = t;
                            break;
                    }
                }
            }
            grid.Children.Add(canvas);

            //獲取當前時間,旋轉對應的角度
            RotateTransform rtf = new RotateTransform();
            rtf.CenterX = CenterPixToX;
            rtf.CenterY = CenterPixToY;
            rtf.Angle = AngleTime;
            canvas.RenderTransform = rtf;
        }

        /// <summary>
        /// 渲染時鐘
        /// </summary>
        public void Show()
        {
            Dispose();
            //設置圓角
            bor.CornerRadius = new CornerRadius(this.ActualWidth / 2);

            Add(AddType.Hour);
            Add(AddType.Minute);
            Add(AddType.Second);

            AddName();

            thread = new Thread(new ThreadStart(threadMethod));
            thread.IsBackground = true;
            thread.Start();
        }

        /// <summary>
        /// 生成名稱
        /// </summary>
        private void AddName()
        {
            TextBlock tb = new TextBlock();
            tb.Text = "XL";
            tb.Foreground = new SolidColorBrush(Colors.White);
            tb.FontSize = 60;
            tb.FontFamily = new FontFamily("華文琥珀");
            tb.HorizontalAlignment = HorizontalAlignment.Center;
            tb.VerticalAlignment = VerticalAlignment.Center;
            grid.Children.Add(tb);
        }

        /// <summary>
        /// UI更新線程
        /// </summary>
        private void threadMethod()
        {
            while (true)
            {
                Dispatcher.Invoke(() =>
                {
                    var s = DateTime.Now.Second;
                    var m = DateTime.Now.Minute;
                    var h = DateTime.Now.Hour;

                    //處理時
                    if (m == 0 && (int)BlockHour.Tag != h)
                    {
                        SetUI(CanvasHour, h);
                    }
                    //處理分
                    if (s == 0 && (int)BlockMinute.Tag != m)
                    {
                        SetUI(CanvasMinute, m);
                    }
                    //處理秒
                    SetUI(CanvasSecond, s);
                });
                Thread.Sleep(1000);
            }
        }

        /// <summary>
        /// 更新UI
        /// </summary>
        /// <param name="can"></param>
        /// <param name="tag"></param>
        /// <param name="color"></param>
        private void SetUI(Canvas can, int tag)
        {
            var type = (AddType)can.Tag;
            foreach (TextBlock item in can.Children)
            {
                if ((int)item.Tag == tag)
                {
                    Debug.WriteLine(item.Text);

                    var fr = item.RenderTransform as RotateTransform;
                    var angle = 360 - fr.Angle;
                    var rtf = can.RenderTransform as RotateTransform;
                    DoubleAnimation db = null;
                    if (type == AddType.Minute)
                    {
                        angle -= 360;
                        db = new DoubleAnimation(angle, new Duration(TimeSpan.FromSeconds(1)));
                        db.Completed += DbMinute_Completed;
                        BlockMinute = item;
                    }
                    else if (type == AddType.Hour)
                    {
                        angle += 360;
                        db = new DoubleAnimation(angle, new Duration(TimeSpan.FromSeconds(1.5)));
                        db.Completed += DbHour_Completed;
                        BlockHour = item;
                    }
                    else
                    {
                        db = new DoubleAnimation(angle, new Duration(TimeSpan.FromSeconds(0.25)));
                        db.Completed += DbSecond_Completed;
                        BlockSecond = item;
                    }
                    rtf.BeginAnimation(RotateTransform.AngleProperty, db);

                }
                else
                {
                    item.Foreground = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#7d7d7d"));
                }
            }
        }

        /// <summary>
        /// 秒 動畫完成時
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void DbSecond_Completed(object sender, EventArgs e)
        {
            BlockSecond.Foreground = new SolidColorBrush(Colors.White);
        }

        /// <summary>
        /// 時 動畫完成時
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void DbHour_Completed(object sender, EventArgs e)
        {
            var fr = CanvasHour.RenderTransform as RotateTransform;
            var angle = fr.Angle - 360;
            fr = null;
            RotateTransform rtf = new RotateTransform();
            rtf.CenterX = CenterPixToX;
            rtf.CenterY = CenterPixToY;
            rtf.Angle = angle;
            CanvasHour.RenderTransform = rtf;
            Debug.WriteLine(rtf.Angle);
            BlockHour.Foreground = new SolidColorBrush(Colors.White);
        }

        /// <summary>
        /// 分 動畫完成時
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void DbMinute_Completed(object sender, EventArgs e)
        {
            var fr = CanvasMinute.RenderTransform as RotateTransform;
            var angle = fr.Angle + 360;
            fr = null;
            RotateTransform rtf = new RotateTransform();
            rtf.CenterX = CenterPixToX;
            rtf.CenterY = CenterPixToY;
            rtf.Angle = angle;
            CanvasMinute.RenderTransform = rtf;
            Debug.WriteLine(rtf.Angle);
            BlockMinute.Foreground = new SolidColorBrush(Colors.White);
        }

        /// <summary>
        /// 釋放
        /// </summary>
        public void Dispose()
        {
            thread?.Abort();
            grid.Children.Clear();
        }
    }

    /// <summary>
    /// 添加類型
    /// </summary>
    public enum AddType
    {
        Hour,
        Minute,
        Second
    }
}

  

  上面是後端邏輯,這才是重點,調用者通過show,調用顯示的;在show裡面會開啟一個後臺處理線程,來實現獲取當前時間,並計算需要旋轉的角度,最後採用動畫更新到UI!

  整個流程就是這樣,有疑問的朋友,歡迎留言!

 

下載地址,附源碼 ==》 點擊我前往

 

支持原創,轉載請標明出處,謝謝!

 

 

  


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一 什麼是消息隊列 我們可以把消息隊列比作是一個存放消息的容器,當我們需要使用消息的時候可以取出消息供自己使用。消息隊列是分散式系統中重要的組件,使用消息隊列主要是為了通過非同步處理提高系統性能和削峰、降低系統耦合性。目前使用較多的消息隊列有ActiveMQ,RabbitMQ,Kafka,Rocket ...
  • 基於flask的網頁聊天室(二) 前言 接上一次的內容繼續完善,今天完成的內容不是很多,只是簡單的用戶註冊登錄,內容具體如下 具體內容 這次要加入與數據哭交互的操作,所以首先要建立相關表結構,這裡使用flask-sqlalchemy來輔助創建 首先修改之前的init文件為: from flask i ...
  • 5.10自我總結 1.編碼解碼 1.字元串編碼 字元 》翻譯過程 》數字 2.字元串解碼 字元 》翻譯過程 》數字 3.編碼解碼用到的翻譯工具 中國:GBK 外國:日本—Shift_JIS,美國ASCII,南韓Euc kr 國際統一:Unicode進行編寫,存取用UTF 8,Unicode與UTF ...
  • 一個業務模塊,是負責完成一系列功能的,這些功能相互之間具有密切的關聯性,所以對於一個模塊來說,業務服務是一個整體,不應把他們再按單個實體拆分開來。OSharp 的業務模塊代碼結構設計,也是根據這一原則來設計的。設計規則如下 ...
  • 1.數字分隔符(Digit Separators ) 數字分隔符使代碼更具可讀性。在聲明變數時,可以將_添加到單獨的數字中。編譯器只刪除_。以下代碼片段在C#7中看起來更具可讀性: In C# 6 In C# 7 In C# 7.2 2.二進位(Binary Literals ) C#7為二進位文件 ...
  • 如果你有一臺Linux伺服器(CentOS7+ 或者 Ubuntu 16.04+)可以使用以下命令快速搭建一個博客。 ...
  • 當月第一天0時0分0秒DateTime.Now.AddDays(1 - DateTime.Now.Day).Date當月最後一天23時59分59秒DateTime.Now.AddDays(1 - DateTime.Now.Day).Date.AddMonths(1).AddSeconds(-1)取當... ...
  • 上一篇博客已經完成 C#ModBus Tcp Master的實現 本篇主要對不同的功能碼所發出的報文進行解析(包括請求報文及響應報文) 讀操作 功能碼 0x01 讀一組線圈 讀取站號為1 從地址12開始的10個線圈 測試結果 上一篇博客已經總結了讀操作報文格式,就按照格式劃分 報文解析: 註意:報文 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...