0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

生成AIに関する記事を書こう!
Qiita Engineer Festa20242024年7月17日まで開催中!

「ClaudeのGPT-4o超え」って意外と本当かもと驚いた!!!

Last updated at Posted at 2024-07-02

image.png
あっという間に、抽選ルーレットが出来上がってしまった話です。

サマリー

この記事でやっていることのサマリーです

  • やりたいことをプロンプトで伝える
  • 変更点をプロンプトで伝える
  • バグをプロンプトで伝える
  • 微調整をプロンプトで伝える

特に生成されたコードがその場でリアルタイムに実行され、出来栄えを見ながらプロンプトに指示することができるのは本当に便利でした。

はじめに

みなさん!話題のClaude使ってますか?Qiita民なら誰よりも早く使ってますよね。
そんな私も『GPT-4oを超えた』とか聞いた時は、「本当かな?」と思い、すぐに試してみて、「やるね、Claude!」と思ったものでした。

が、しかし、今回発表されたClaude 3.5 Sonnet Artifactsには本当に驚いてしまいました。
「何をそんなに驚いているの?」って聞こえてきそうですが、まずは、こちらの動画をご覧ください。

実は、これ、Claude 3.5 Sonnet Artifactsに236文字のプロンプトを入力しただけで生成されたアプリなんです。

準備

準備といっても、

  1. Claudeのホームページにアクセス
  2. Googleアカウントでログイン
  3. 表示されたプロンプト入力欄にプロンプトを入力する
    image.png

これだけです。

レッツ・トライ!

それでは、早速、私が入力したプロンプトを公開いたします。
コピペもOKです:bangbang:

プロンプト 1回目
プレゼンをする人を決定するアプリを作成してください。プレゼン者を入力するフィールドもつけてください

出力結果は、こちら
image.png
プログラムが生成されるだけでなく、生成されたプログラムが実際に実行され、右側に表示されます。
プロンプトを入力してから、生成されたプログラムが動くまでの様子をこの動画でご確認ください。

なんと、一発で動くプログラムが生成されましたし、そのプログラムがClaude上でリアルタイムに動作し確認まで出来ます (すごい)

でも、なんかちょっと寂しいですね (贅沢)
せっかくなんで、抽選の雰囲気を出すために、ルーレット風に変えてみます。

プロンプト 2回目
ルーレットを回すことで決まるようなアプリにしてください

サクッと修正が入りました。こんな感じです。
image.png
それでは実際に動かしてみます。

ルーレットが回ったところまでは良かったのですが、結果は、ちょうど境目で止まって、どちらが選ばれたのか、判断が微妙です。
image.png
プログラム的には『宮崎』さんです。

プロンプト 3回目
ちょうど境目で止まり、プレゼン発表者を決めることができませんけど

また、プログラムの再生成が始まりました。
実行してみると、前回と同じように、また境目で止まり、プレゼン発表者を決めることができません。
仕方ないので、もう一度。ちょっとプロンプトを変えてみます。

プロンプト 4回目
まだ矢印が境目で止まります。境目で止まらないようにしてください

またまた、プログラムの再生成が始まりました。
今度は良さそう:interrobang:
境目のところで止まらなくなったのですが、今度は、矢印と選ばれたプレゼン者が違いますね。もう!
image.png

また修正プロンプトを入力してみます。

プロンプト 5回目
矢印が止まった箇所と選ばれたプレゼン者が違いますよ

今度は、矢印の先と選ばれたプレゼン者が一致しました。

image.png
欲を言えば、ルーレット内のフォントが小さくて見づらいですね。

プロンプト 6回目
入力したプレゼン者のフォントの大きさを2倍にしてください

ルーレット内の文字のフォントが大きくなり、見やすいです:bangbang:

image.png
完成したと思いきや、実は数回[ルーレットを回す]をクリックしてみると、いつも同じ人が選ばれてしまいます。
(わたくし、柳ばかりが選ばれます。これはいかん、いかん)

プロンプト 7回目
いつも同じ人が選ばれてしまいます。回転速度をランダムに変えてください

ランダムにプレゼン者が選ばれるようです。ついに完成しました:bangbang::bangbang:

スゴすぎる:grin:

参考: 生成されたソースコード

node.js
import React, { useState, useEffect, useRef } from 'react';
import { Button } from '@/components/ui/button';
import { Input } from '@/components/ui/input';

const PresenterRoulette = () => {
  const [presenters, setPresenters] = useState([]);
  const [newPresenter, setNewPresenter] = useState('');
  const [selectedPresenter, setSelectedPresenter] = useState(null);
  const [isSpinning, setIsSpinning] = useState(false);
  const canvasRef = useRef(null);

  const addPresenter = () => {
    if (newPresenter.trim() !== '') {
      setPresenters([...presenters, newPresenter.trim()]);
      setNewPresenter('');
    }
  };

  const spinRoulette = () => {
    if (presenters.length > 0 && !isSpinning) {
      setIsSpinning(true);
      const minSpins = 5;
      const maxSpins = 10;
      const spins = minSpins + Math.random() * (maxSpins - minSpins);
      const minDuration = 4000;
      const maxDuration = 8000;
      const duration = minDuration + Math.random() * (maxDuration - minDuration);
      let currentRotation = 0;
      const startTime = Date.now();

      const sliceAngle = 360 / presenters.length;
      const additionalRotation = Math.random() * 360; // 完全にランダムな追加回転

      const animate = () => {
        const elapsedTime = Date.now() - startTime;
        const progress = Math.min(elapsedTime / duration, 1);
        const easeOut = 1 - Math.pow(1 - progress, 3);
        
        currentRotation = easeOut * spins * 360 + additionalRotation;
        drawRoulette(currentRotation);

        if (progress < 1) {
          requestAnimationFrame(animate);
        } else {
          setIsSpinning(false);
          selectPresenter(currentRotation);
        }
      };

      animate();
    }
  };

  const selectPresenter = (finalRotation) => {
    const normalizedRotation = finalRotation % 360;
    const sliceAngle = 360 / presenters.length;
    const selectedIndex = (presenters.length - Math.floor(normalizedRotation / sliceAngle) - 1) % presenters.length;
    setSelectedPresenter(presenters[selectedIndex]);
  };

  const drawRoulette = (rotation) => {
    const canvas = canvasRef.current;
    const ctx = canvas.getContext('2d');
    const centerX = canvas.width / 2;
    const centerY = canvas.height / 2;
    const radius = Math.min(centerX, centerY) - 10;

    ctx.clearRect(0, 0, canvas.width, canvas.height);
    ctx.save();
    ctx.translate(centerX, centerY);
    ctx.rotate((rotation * Math.PI) / 180);

    const sliceAngle = (2 * Math.PI) / presenters.length;
    const colors = ['#FF6B6B', '#4ECDC4', '#45B7D1', '#FFA07A', '#98D8C8'];

    presenters.forEach((presenter, index) => {
      const startAngle = index * sliceAngle;
      const endAngle = startAngle + sliceAngle;

      ctx.beginPath();
      ctx.moveTo(0, 0);
      ctx.arc(0, 0, radius, startAngle, endAngle);
      ctx.closePath();
      ctx.fillStyle = colors[index % colors.length];
      ctx.fill();

      ctx.save();
      ctx.rotate(startAngle + sliceAngle / 2);
      ctx.textAlign = 'right';
      ctx.fillStyle = '#000';
      ctx.font = 'bold 28px Arial';
      ctx.fillText(presenter, radius - 30, 0);
      ctx.restore();
    });

    ctx.restore();

    // Draw arrow
    ctx.beginPath();
    ctx.moveTo(centerX + radius, centerY);
    ctx.lineTo(centerX + radius + 20, centerY - 10);
    ctx.lineTo(centerX + radius + 20, centerY + 10);
    ctx.closePath();
    ctx.fillStyle = '#333';
    ctx.fill();
  };

  useEffect(() => {
    if (canvasRef.current) {
      drawRoulette(0);
    }
  }, [presenters]);

  return (
    <div className="p-4 max-w-md mx-auto">
      <h1 className="text-2xl font-bold mb-4">プレゼン者ルーレットアプリ</h1>
      <div className="flex mb-4">
        <Input
          type="text"
          value={newPresenter}
          onChange={(e) => setNewPresenter(e.target.value)}
          placeholder="プレゼン者の名前を入力"
          className="flex-grow mr-2"
        />
        <Button onClick={addPresenter}>追加</Button>
      </div>
      <div className="mb-4">
        <h2 className="text-lg font-semibold mb-2">プレゼン者リスト:</h2>
        <ul className="list-disc pl-5">
          {presenters.map((presenter, index) => (
            <li key={index}>{presenter}</li>
          ))}
        </ul>
      </div>
      <div className="mb-4">
        <canvas ref={canvasRef} width="300" height="300" className="mx-auto"></canvas>
      </div>
      <Button onClick={spinRoulette} className="w-full mb-4" disabled={isSpinning}>
        {isSpinning ? 'ルーレット回転中...' : 'ルーレットを回す'}
      </Button>
      {selectedPresenter && (
        <div className="text-center p-4 bg-blue-100 rounded">
          <h2 className="text-xl font-bold">選ばれたプレゼン者:</h2>
          <p className="text-2xl text-blue-600">{selectedPresenter}</p>
        </div>
      )}
    </div>
  );
};

export default PresenterRoulette;

終わりに

GPT-4o超えとか、最初は信じられませんでしたが、今回、やりたい事を素直にプロンプトに入力したら、たった10分程度で、こんなアプリがサクッとできてしまいました。
「Claudeが本当にGPT-4oを超えてしまったかもしれない:laughing:

参考

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?